QOJ.ac
QOJ
ID | Problem | Submitter | Result | Time | Memory | Language | File size | Submit time | Judge time |
---|---|---|---|---|---|---|---|---|---|
#683119 | #9524. 1D Galaxy | ucup-team3586# | WA | 1ms | 9776kb | C++23 | 7.7kb | 2024-10-27 18:50:47 | 2024-10-27 18:50:49 |
Judging History
answer
//Author: Kevin
#include<bits/stdc++.h>
//#pragma GCC optimize("O2")
using namespace std;
#define ll long long
#define ull unsigned ll
#define pb emplace_back
#define mp make_pair
#define ALL(x) (x).begin(),(x).end()
#define rALL(x) (x).rbegin(),(x).rend()
#define srt(x) sort(ALL(x))
#define rev(x) reverse(ALL(x))
#define rsrt(x) sort(rALL(x))
#define sz(x) (int)(x.size())
#define inf 0x3f3f3f3f
#define pii pair<int,int>
#define lb(v,x) (int)(lower_bound(ALL(v),x)-v.begin())
#define ub(v,x) (int)(upper_bound(ALL(v),x)-v.begin())
#define uni(v) v.resize(unique(ALL(v))-v.begin())
#define longer __int128_t
void die(string S){puts(S.c_str());exit(0);}
mt19937_64 rng(114514);
int n,q;
ll x[100100],w[100100],s[100100];
ll mn[100100],mx[100100];
struct query
{
ll t;
int ind,x;
bool friend operator <(const query &a,const query &b){return a.t<b.t;}
};
vector<query> vec;
ull prior[100100];
int ls[100100],rs[100100];
int tag[100100];
int siz[100100];
int fa[100100];
void pushdown(int u)
{
x[u]+=tag[u];
mn[u]+=tag[u];
mx[u]+=tag[u];
if(ls[u]) tag[ls[u]]+=tag[u];
if(rs[u]) tag[rs[u]]+=tag[u];
tag[u]=0;
}
void pushup(int x)
{
pushdown(x);
if(ls[x]) pushdown(ls[x]);
if(rs[x]) pushdown(rs[x]);
s[x]=s[ls[x]]+s[rs[x]]+w[x];
siz[x]=siz[ls[x]]+siz[rs[x]]+1;
mn[x]=(ls[x]?mn[ls[x]]:(::x[x]));
mx[x]=(rs[x]?mx[rs[x]]:(::x[x]));
fa[ls[x]]=fa[rs[x]]=x;
}
int merge(int u,int v)
{
if(!u||!v)
{
pushup(u+v);
return u+v;
}
if(prior[u]>prior[v])
{
pushdown(u);
rs[u]=merge(rs[u],v);
pushup(u);
return u;
}
else
{
pushdown(v);
ls[v]=merge(u,ls[v]);
pushup(v);
return v;
}
}
void splits(int u,int s,int &a,int &b)
{
if(!u)
{
a=b=0;
return ;
}
pushdown(u);
if(siz[u]+1<=s)
{
a=u;
splits(rs[u],s-siz[u]-1,rs[a],b);
pushup(a);
return ;
}
else
{
b=u;
splits(ls[u],s,a,ls[b]);
pushup(b);
return ;
}
}
void splitv(int u,ll v,int &a,int &b)
{
if(!u)
{
a=b=0;
return ;
}
pushdown(u);
if(s[ls[u]]+w[u]<=v)
{
a=u;
splitv(rs[u],v-w[u]-s[ls[u]],rs[a],b);
pushup(a);
return ;
}
else
{
b=u;
splitv(ls[u],v,a,ls[b]);
pushup(b);
return ;
}
}
void splitx(int u,int p,int &a,int &b)
{
if(!u)
{
a=b=0;
return ;
}
pushdown(u);
if(x[u]<=p)
{
a=u;
splitx(rs[u],p,rs[a],b);
pushup(a);
return ;
}
else
{
b=u;
splitx(ls[u],p,a,ls[b]);
pushup(b);
return ;
}
}
void pushdown_all(int x)
{
pushdown(x);
if(ls[x]) pushdown_all(ls[x]);
if(rs[x]) pushdown_all(rs[x]);
}
int getx(int ind)
{
int res=x[ind];
while(true)
{
res+=tag[ind];
if(ind==ls[fa[ind]]||ind==rs[fa[ind]])
ind=fa[ind];
else
break;
}
return res;
}
int ans[100100];
int main()
{
ios_base::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
cin>>n>>q;
for(int i=1;i<=n;i++)
cin>>x[i]>>w[i];
mn[0]=inf;
mx[0]=-inf;
for(int i=1;i<=q;i++)
{
query tmp;
cin>>tmp.t>>tmp.x;
tmp.ind=i;
vec.pb(tmp);
}
srt(vec);
int rt=0;
for(int i=1;i<=n;i++)
{
prior[i]=rng();
siz[i]=1;
s[i]=w[i];
mn[i]=mx[i]=x[i];
int A,B;
splitx(rt,x[i],A,B);
rt=merge(A,merge(i,B));
}
ll cur=0;
int qp=0;
ll total=accumulate(w+1,w+n+1,0ll);
while(true)
{
// cerr<<"new round"<<endl;
// cerr<<mn[rt]<<" "<<mx[rt]<<endl;
if(mn[rt]==mx[rt])
{
// cerr<<"Type 0-1"<<endl;
while(qp<sz(vec))
{
ans[vec[qp].ind]=x[rt];
qp++;
}
break;
}
if(mn[rt]==mx[rt]-1)
{
// cerr<<"Type 0-2"<<endl;
pushdown_all(rt);
while(qp<sz(vec))
{
int sind=vec[qp].x;
int ori=x[sind];
if((vec[qp].t-cur)&1)
ori=mn[rt]+mx[rt]-ori;
ans[vec[qp].ind]=ori;
qp++;
}
break;
}
ll half=total/2;
int A,B,C;
int ppos;
splitv(rt,half,A,C);
ppos=mx[A];
if(mx[A]==mn[C]) ppos--;
// cerr<<ppos<<endl;
rt=merge(A,C);
splitx(rt,ppos,A,B);
splitx(B,mn[B],B,C);
// cerr<<A<<" "<<B<<" "<<C<<endl;
if(s[A]==s[C])
{
// cerr<<"Type 1"<<endl;
ll need=min(x[B]-mx[A],mn[C]-x[B]);
while(qp<sz(vec)&&vec[qp].t<=cur+need)
{
int sind=vec[qp].x;
int qind=vec[qp].ind;
ll T=vec[qp].t-cur;
int X=getx(sind);
if(X==x[B])
ans[qind]=X;
else if(X<x[B])
ans[qind]=X+T;
else
ans[qind]=X-T;
qp++;
}
cur+=need;
tag[A]+=need;
tag[C]-=need;
rt=merge(A,merge(B,C));
}
else
{
if(s[A]>s[C])
C=merge(B,C);
else
A=merge(A,B);
ll X1=mx[A];
ll X2=mn[C];
if((X2-X1)%2==0)
{
// cerr<<"Type 2"<<endl;
// cerr<<A<<" "<<C<<" "<<ls[A]<<" "<<rs[A]<<" "<<ls[C]<<" "<<rs[C]<<endl;
// cerr<<cur<<" "<<mx[A]<<" "<<mn[C]<<endl;
ll need=(X2-X1)/2;
while(qp<sz(vec)&&vec[qp].t<=cur+need)
{
int sind=vec[qp].x;
int qind=vec[qp].ind;
ll T=vec[qp].t-cur;
int X=getx(sind);
if(X<=X1)
ans[qind]=X+T;
else
ans[qind]=X-T;
qp++;
}
cur+=need;
tag[A]+=need;
tag[C]-=need;
rt=merge(A,C);
}
else
{
// cerr<<"Type 3"<<endl;
int S1,S2,S3,S4;
ll need=(X2-X1)/2;
while(qp<sz(vec)&&vec[qp].t<=cur+need)
{
int sind=vec[qp].x;
int qind=vec[qp].ind;
ll T=vec[qp].t-cur;
int X=getx(sind);
if(X<=X1)
ans[qind]=X+T;
else
ans[qind]=X-T;
qp++;
}
cur+=need;
tag[A]+=need;
tag[C]-=need;
pushdown(A);
pushdown(C);
splitx(A,mx[A]-1,S1,S2);
splitx(C,mn[C],S3,S4);
if(mx[S1]==mx[S2]-1||mn[S3]+1==mn[S4])
{
// cerr<<"Type 3-1"<<endl;
// cerr<<S1<<" "<<S2<<" "<<S3<<" "<<S4<<endl;
// cerr<<ls[S2]<<" "<<rs[S2]<<endl;
tag[S1]++;
tag[S4]--;
rt=merge(S1,S4);
tag[S2]++;
tag[S3]--;
pushdown(S2);
pushdown(S3);
int tmpA,tmpB;
splitx(rt,mn[S2],tmpA,tmpB);
rt=merge(tmpA,merge(S2,tmpB));
splitx(rt,mn[S3],tmpA,tmpB);
rt=merge(tmpA,merge(S3,tmpB));
cur++;
while(qp<sz(vec)&&vec[qp].t==cur)
{
int sind=vec[qp].x;
int qind=vec[qp].ind;
ans[qind]=getx(sind);
qp++;
}
}
else
{
if(s[S1]>=s[S2]+s[S4]||s[S4]>=s[S1]+s[S3])
{
// cerr<<"Type 3-2"<<endl;
tag[S1]++;
tag[S4]--;
rt=merge(S1,S4);
tag[S2]++;
tag[S3]--;
pushdown(S2);
pushdown(S3);
int tmpA,tmpB;
splitx(rt,mn[S2],tmpA,tmpB);
rt=merge(tmpA,merge(S2,tmpB));
splitx(rt,mn[S3],tmpA,tmpB);
rt=merge(tmpA,merge(S3,tmpB));
cur++;
while(qp<sz(vec)&&vec[qp].t==cur)
{
int sind=vec[qp].x;
int qind=vec[qp].ind;
ans[qind]=getx(sind);
qp++;
}
continue;
}
// cerr<<"Type 3-3"<<endl;
ll need=min(mx[S2]-mx[S1],mn[S4]-mn[S3]);
int XX1=x[S2],XX2=x[S3];
while(qp<sz(vec)&&vec[qp].t<=cur+need)
{
int sind=vec[qp].x;
int qind=vec[qp].ind;
ll T=vec[qp].t-cur;
int X=getx(sind);
if(X<XX1)
ans[qind]=X+T;
else if(X>XX2)
ans[qind]=X-T;
else if(X==XX1)
ans[qind]=(T&1)?XX2:XX1;
else
ans[qind]=(T&1)?XX1:XX2;
qp++;
}
cur+=need;
if(need&1)
{
tag[S2]++;
tag[S3]--;
}
tag[S1]+=need;
tag[S4]-=need;
rt=merge(S1,S4);
pushdown(S2);
pushdown(S3);
int tmpA,tmpB;
splitx(rt,mn[S2],tmpA,tmpB);
rt=merge(tmpA,merge(S2,tmpB));
splitx(rt,mn[S3],tmpA,tmpB);
rt=merge(tmpA,merge(S3,tmpB));
}
}
}
}
for(int i=1;i<=q;i++)
cout<<ans[i]<<'\n';
return 0;
}
Details
Tip: Click on the bar to expand more detailed information
Test #1:
score: 100
Accepted
time: 1ms
memory: 7676kb
input:
4 12 0 1 1 2 -1 3 2 2 0 1 0 2 0 3 0 4 1 1 1 2 1 3 1 4 2 1 2 2 2 3 2 4
output:
0 1 -1 2 1 0 0 1 0 1 1 0
result:
ok 12 lines
Test #2:
score: 0
Accepted
time: 1ms
memory: 9756kb
input:
5 5 -4 5 -1 3 5 5 0 4 -4 2 0 3 3 1 5 5 1 3 3 3
output:
5 -1 1 4 2
result:
ok 5 lines
Test #3:
score: 0
Accepted
time: 1ms
memory: 9776kb
input:
5 5 3 2 1 3 0 3 1 5 0 5 7 1 1 4 1 5 8 5 10 3
output:
0 0 1 0 0
result:
ok 5 lines
Test #4:
score: 0
Accepted
time: 1ms
memory: 7684kb
input:
5 5 -3 2 -2 1 5 3 0 1 1 5 10 5 3 1 1 3 2 3 8 4
output:
1 0 4 3 0
result:
ok 5 lines
Test #5:
score: 0
Accepted
time: 1ms
memory: 7772kb
input:
5 5 0 5 -3 1 -2 5 -5 2 -3 4 6 1 7 3 2 3 6 1 8 1
output:
-2 -3 -2 -2 -2
result:
ok 5 lines
Test #6:
score: 0
Accepted
time: 0ms
memory: 7772kb
input:
5 5 -1 1 -8 4 0 2 4 3 5 1 1 3 10 5 4 4 4 1 9 4
output:
-1 -1 0 -1 -1
result:
ok 5 lines
Test #7:
score: 0
Accepted
time: 1ms
memory: 9704kb
input:
4 4 -5 4 1 0 2 3 -1 -2 1 1 1 3 0 1 9 2
output:
-4 1 -5 -2
result:
ok 4 lines
Test #8:
score: -100
Wrong Answer
time: 1ms
memory: 7832kb
input:
4 4 -7 -2 10 1 9 2 -10 3 11 3 4 3 0 1 20 4
output:
-1 6 -7 0
result:
wrong answer 1st lines differ - expected: '0', found: '-1'