제출 #1130069

#제출 시각아이디문제언어결과실행 시간메모리
1130069Math4Life2020Two Currencies (JOI23_currencies)C++20
38 / 100
5138 ms1008084 KiB
#pragma GCC optimize("Ofast,unroll-loops") #pragma GCC target("avx2") #include <bits/stdc++.h> using namespace std; using ll = long long; using pii = pair<ll,ll>; const int Nm = 1e5+5; const int E = 17; const int INF = 1e9; struct Node { //map<ll,ll> m0; //cost->number //map<ll,pii> mps; //aggregate: cost -> {# <=cost, val <=cost} //vector<pair<int,int>> m0; vector<pair<pair<int,int>,long long>> mps; int ncp = 0; bool brk = 0; void prc(vector<pair<int,int>> m0) { int n0=0; long long v0=0; //mps[-INF]={0,0}; mps.push_back({{-INF,0},0LL}); for (pair<int,int> p0: m0) { long long c1 = p0.first; int n1 = p0.second; n0+=n1; v0+=c1*n1; //mps[c1]={n0,v0}; mps.push_back({{c1,n0},v0}); } } vector<pair<int,int>> getm0() { vector<pair<int,int>> vf; int ntot = 0; long long vtot = 0; for (auto A0: mps) { if (A0.first.first==(-INF)) { continue; } int n1 = A0.first.second; long long c1 = A0.second; vf.push_back({(c1-vtot)/(n1-ntot),n1-ntot}); ntot = n1; vtot = c1; } return vf; } Node(){brk=1;} Node(vector<ll> v1) { map<ll,ll> mtest; for (ll x: v1) { mtest[x]++; } ncp = v1.size(); vector<pair<int,int>> m0; for (pii p0: mtest) { m0.push_back(p0); } prc(m0); } Node(Node n1, Node n2) { map<int,int> mtest; vector<pair<int,int>> n1m0 = n1.getm0(); for (pair<int,int> p0: n1m0) { mtest[p0.first]+=p0.second; } n1m0 = n2.getm0(); for (pair<int,int> p0: n1m0) { mtest[p0.first]+=p0.second; } ncp = n1.ncp+n2.ncp; vector<pair<int,int>> m0; for (pair<int,int> p0: mtest) { m0.push_back(p0); } prc(m0); } pii qry(int x) { auto A0 = --upper_bound(mps.begin(),mps.end(),(pair<pair<int,int>,long long>){{x,INF},INF}); return {(*A0).first.second,(*A0).second}; } }; int bjump[Nm][E]; Node ejump[Nm][E]; pair<int,int> ett[2*Nm][E+1]; vector<pii> adj[Nm]; vector<ll> fadj[Nm]; ll radj[Nm]; ll locr[Nm]; //location of road (aka location of child) ll ht[Nm]; ll floc[Nm]; inline ll gp(ll x, ll e) { if (x<0) { return x; } return bjump[x][e]; } inline pii fz(pii p1, pii p2) { if (p1.second<p2.second) { return p1; } else { return p2; } } inline ll l2(ll x) { return (31-__builtin_clz(x)); } inline ll v2(ll x) { return __builtin_ctz(x); } ll lca(ll x, ll y) { x = floc[x]; y = floc[y]; if (x>y) { swap(x,y); } ll lv = l2(y-x+1); return (fz(ett[x][lv],ett[y-(1<<lv)+1][lv]).first); } pii ginf(vector<pii>& v1, ll cst) { pii vf = {0,0}; for (pii A0: v1) { pii v2 = ejump[A0.first][A0.second].qry(cst); vf = {vf.first+v2.first,vf.second+v2.second}; } return vf; //{number, total cost} } ll qry(vector<pii>& v1, ll y) { ll xmin = 0; ll xmax = 1e9; //first binary search for the largest value of xtest //for which ginf(xtest).second <= y //store pair at this value while (xmin < xmax) { ll xtest = (xmin+xmax+1)/2; if ((ginf(v1,xtest).second)<=y) { xmin = xtest; } else { xmax = xtest-1; } } pii p1 = ginf(v1,xmin); pii p2 = ginf(v1,xmin+1); if (p1.first==p2.first || p1.second==y) { return p1.first; } else { ll vext = y-p1.second; ll cst = (p2.second-p1.second)/(p2.first-p1.first); //assert(cst==(xmin+1)); return (p1.first+vext/cst); } } int main() { ios_base::sync_with_stdio(false); cin.tie(0); ll N,M,Q; cin >> N >> M >> Q; for (ll i=0;i<(N-1);i++) { ll a,b; cin >> a >> b; a--; b--; adj[a].push_back({b,i}); adj[b].push_back({a,i}); } radj[0]=-1; ht[0]=0; queue<ll> q0; q0.push(0); while (!q0.empty()) { ll x = q0.front(); q0.pop(); for (pii p0: adj[x]) { ll y = p0.first; if (y != radj[x]) { fadj[x].push_back(y); radj[y]=x; locr[p0.second]=y; q0.push(y); ht[y]=1+ht[x]; } } } vector<ll> ettv; stack<pii> st0; st0.push({0,0}); while (!st0.empty()) { pii p0 = st0.top(); st0.pop(); ll x = p0.first; ll t = p0.second; ettv.push_back(x); if (t==0) { floc[x]=ettv.size()-1; for (ll y: fadj[x]) { st0.push({x,1}); st0.push({y,0}); } } } for (ll i=0;i<ettv.size();i++) { ett[i][0]={ettv[i],ht[ettv[i]]}; } for (ll e=1;e<=E;e++) { for (ll i=0;i<=(2*Nm-(1<<e));i++) { ett[i][e]=fz(ett[i][e-1],ett[i+(1<<(e-1))][e-1]); } } vector<ll> costE[Nm]; for (ll m=0;m<M;m++) { ll p,c; cin >> p >> c; p--; costE[locr[p]].push_back(c); } for (ll i=0;i<N;i++) { bjump[i][0]=radj[i]; } for (ll i=0;i<N;i++) { ejump[i][0]=Node(costE[i]); } for (ll e=1;e<E;e++) { for (ll i=0;i<N;i++) { if (bjump[i][e-1]==-1) { bjump[i][e]=bjump[i][e-1]; //ejump[i][e]=Node(); } else { bjump[i][e]=bjump[bjump[i][e-1]][e-1]; if (!ejump[bjump[i][e-1]][e-1].brk && e<=15) ejump[i][e]=Node(ejump[i][e-1],ejump[bjump[i][e-1]][e-1]); } } } for (ll q=0;q<Q;q++) { ll s,t,x,y; cin >> s >> t >> x >> y; s--; t--; if (s==t) { cout << x << "\n"; continue; } //exit(0); ll lc = lca(s,t); //cout << "lc="<<lc<<"\n"; exit(0); //cout << "lca of s,t="<<s<<","<<t<<" is "<<lc<<"\n"; //ll dist0 = ht[s]+ht[t]-2*ht[lc]; vector<pii> vq; ll ntoll = 0; while (s != lc) { ll dht = ht[s]-ht[lc]; ll ldh = l2(dht); ldh = min((ll)ldh,15LL); vq.push_back({s,ldh}); ntoll += ejump[s][ldh].ncp; s=bjump[s][ldh]; } while (t != lc) { ll dht = ht[t]-ht[lc]; ll ldh = l2(dht); ldh = min((ll)ldh,15LL); vq.push_back({t,ldh}); ntoll += ejump[t][ldh].ncp; t=bjump[t][ldh]; } ll maxr = qry(vq,y); ll gused = max(0LL,ntoll-maxr); //cout << "ntoll,maxr="<<ntoll<<","<<maxr<<"\n"; if (x<gused) { cout << "-1\n"; } else { cout << (x-gused) <<"\n"; } // cout << "maxr="<<maxr<<"\n"; // cout << "dist0="<<dist0<<"\n"; // if ((maxr+x)>=dist0) { // cout << (maxr+x-dist0) <<"\n"; // } else { // cout << "-1\n"; // } } }
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...