이 제출은 이전 버전의 oj.uz에서 채점하였습니다. 현재는 제출 당시와는 다른 서버에서 채점을 하기 때문에, 다시 제출하면 결과가 달라질 수도 있습니다.
//marico el que lo lea
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <vector>
#include <map>
#include <set>
#include <queue>
#include <stack>
#include <stdlib.h>
#include <assert.h>
using namespace std;
typedef long long ll;
typedef vector<int> vi;
typedef vector<ll> vll;
typedef pair<int, int> ii;
void fastIO() {
std::ios_base::sync_with_stdio(0);cin.tie(0);cout.tie(0);
}
#define FOR(i,f,t) for(int i=f; i<(int)t; i++)
#define FORR(i,f,t) for(int i=f; i>(int)t; i--)
#define FORE(i,c) for(auto i = (c).begin(); i != (c).end(); i++)
#define pb push_back
#define all(obj) obj.begin(), obj.end()
#define ms(obj, val) memset(obj, val, sizeof(obj))
#define ms2(obj, val, sz) memset(obj, val, sizeof(obj[0])*sz)
#define fst first
#define snd second
template<typename T, typename U> inline void mnze(T &x, U y) { if(y < x) x = y; }
template<typename T, typename U> inline void mxze(T &x, U y) { if(x < y) x = y; }
void _scan( int &x ) { scanf("%d",&x); }
void _scan( long long &x ) { scanf("%lld",&x); }
void _scan( double &x ) { scanf("%lf",&x); }
void _scan( char &x ) { scanf(" %c",&x); }
void _scan( char *x ) { scanf("%s",x); }
void scan() {}
template<typename T, typename... U>
void scan( T& head, U&... tail ) { _scan(head); scan(tail...);}
template<typename T> void _dbg(const char* sdbg, T h) { cerr<<sdbg<<"="<<h<<"\n"; }
template<typename T, typename... U> void _dbg(const char* sdbg, T h, U... t) {
while(*sdbg != ',')cerr<<*sdbg++; cerr<<"="<<h<<","; _dbg(sdbg+1, t...);
}
#ifdef LOCAL
#define debug(...) _dbg(#__VA_ARGS__, __VA_ARGS__)
#define debugv(x) {{cerr <<#x <<" = "; FORE(_i, (x)) cerr <<*_i <<", "; cerr <<"\n"; }}
#define debuga(x, sz) {{cerr <<#x <<" = "; FOR(_i, 0, sz) cerr << x[_i] <<", "; cerr <<"\n"; }}
#else
#define debug(...) (__VA_ARGS__)
#define debugv(x)
#define debuga(x, sz)
#define cerr if(0)cout
#endif
struct stnode{
int sum;
stnode *l=NULL, *r=NULL;
void merge(){merge(*l, *r);}
void merge(stnode& L, stnode& R){sum = L.sum + R.sum;}
void upd(int x){sum += x;}
};
typedef stnode* pstnode;
const int MAXNODES = 12e6+6;
stnode nodes[MAXNODES];
struct ST{
vector<pstnode> root, all;
vector<stnode> nodes;
int N, ni;
void clear(){
all.clear(); root.clear();
nodes.clear();
N=ni=0;
}
pstnode new_node(){
if(ni+1 == MAXNODES) printf("NOOOO\n");
all.pb(&nodes[ni++]);
all.back()->sum=0;
return all.back();
}
void init(int n){
nodes = vector<stnode>(MAXNODES);
root.pb(new_node());
N = n;
}
void build(){
root[0]->sum = 0;
root[0]->l=root[0];
root[0]->r=root[0];
}
// erase all elements < b in ver (used)
inline int erase(int b, int ver){
root.pb(erase(b, root[ver], root[0], 0, N));
return root.size()-1;
}
inline pstnode erase(int b, pstnode v, pstnode emptyv, int l, int r){
if(r<=b) return emptyv;
int M = (l+r)>>1;
pstnode u = new_node();
if(b<=M){
u->r = v->r;
u->l = erase(b, v->l, emptyv->l, l, M);
}else{
u->l = emptyv->l;
u->r = erase(b, v->r, emptyv->r, M, r);
}
u->merge();
return u;
}
int K;
inline bool bs(int k, int ver, int &usedver){
K = k;
//debug(k);
if(root[ver]->sum - root[usedver]->sum < K) return false;
//debug(root[ver]->sum, root[usedver]->sum);
root.pb( bs(root[ver], root[usedver], 0, N) );
usedver = root.size()-1;
//debug(root[usedver]->sum);
return K==0;
}
// we are always in nodes were we have enough.
inline pstnode bs(pstnode v, pstnode usedv, int l, int r){
int lft = v->sum - usedv->sum;
if(lft == K){
K=0;
return v;
}
pstnode u = new_node();
u->sum = usedv->sum;
if(r-l==1){
u->sum += K;
K=0;
return u;
}
int M = (l+r)>>1;
int llft = v->l->sum - usedv->l->sum;
if(llft >= K){
u->r = usedv->r;
u->l = bs(v->l, usedv->l, l, M);
}else{
K -= llft;
u->l = v->l;
u->r = bs(v->r, usedv->r, M, r);
}
u->merge();
return u;
}
inline int update(int p, int x, int ver){
root.pb(update(p, x, root[ver], 0, N));
return root.size()-1;
}
inline pstnode update(int p, int x, pstnode v, int l, int r){
pstnode u = new_node();
if(r - l < 2){
u->sum = v->sum;
u->upd(x);
return u;
}
int M = (l+r)>>1;
u->l = v->l, u->r = v->r;
if(p < M) u->l = update(p, x, v->l, l, M);
else u->r = update(p, x, v->r, M, r);
u->merge();
return u;
}
};
///////////////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////////////
const int MAXN = 5e5+5;
int N, A[MAXN], B[MAXN], BtoB[MAXN];
vi AtoB[MAXN];
ST st;
vi strt;
int M, K[MAXN];
int solve(){
int ans = 1;
sort(K, K+M);
int ni = st.ni;
//debug(ni);
int usedrt = 0;
FOR(i,0,M && ans){
usedrt = st.erase(K[i], usedrt);
ans &= st.bs(K[i], strt[K[i]], usedrt);
}
st.ni = ni;
return ans;
}
void pre(){
FOR(i,0,N){
AtoB[A[i]].pb(B[i]);
BtoB[B[i]]++;
}
st.init(N+5);
st.build();
strt.pb(0);
FOR(i,1,N+1){
int nxt = strt.back();
//debug(i);
//debug(st.root[nxt]->sum);
for(int b : AtoB[i]) nxt = st.update(b, 1, nxt);
if(BtoB[i-1]) nxt = st.update(i-1, -BtoB[i-1], nxt);
//debug(BtoB[i-1]);
strt.pb(nxt);
//debug(st.root[strt[i]]->sum);
}
}
void init(int NN, int *AA, int *BB){
N=NN;
FOR(i,0,N) A[i]=AA[i];
FOR(i,0,N) B[i]=BB[i];
pre();
}
int can(int MM, int *KK){
M = MM;
FOR(i,0,M) K[i]=KK[i];
return solve();
}
컴파일 시 표준 에러 (stderr) 메시지
teams.cpp: In member function 'int ST::erase(int, int)':
teams.cpp:104:28: warning: conversion to 'int' from 'std::vector<stnode*>::size_type {aka long unsigned int}' may alter its value [-Wconversion]
return root.size()-1;
^
teams.cpp: In member function 'bool ST::bs(int, int, int&)':
teams.cpp:127:17: warning: conversion to 'int' from 'std::vector<stnode*>::size_type {aka long unsigned int}' may alter its value [-Wconversion]
usedver = root.size()-1;
^
teams.cpp: In member function 'int ST::update(int, int, int)':
teams.cpp:162:22: warning: conversion to 'int' from 'std::vector<stnode*>::size_type {aka long unsigned int}' may alter its value [-Wconversion]
return root.size()-1;
^
teams.cpp: In function 'void _scan(int&)':
teams.cpp:39:38: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
void _scan( int &x ) { scanf("%d",&x); }
^
teams.cpp: In function 'void _scan(long long int&)':
teams.cpp:40:46: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
void _scan( long long &x ) { scanf("%lld",&x); }
^
teams.cpp: In function 'void _scan(double&)':
teams.cpp:41:42: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
void _scan( double &x ) { scanf("%lf",&x); }
^
teams.cpp: In function 'void _scan(char&)':
teams.cpp:42:40: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
void _scan( char &x ) { scanf(" %c",&x); }
^
teams.cpp: In function 'void _scan(char*)':
teams.cpp:43:38: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
void _scan( char *x ) { scanf("%s",x); }
^
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |