年前的坑今天補(bǔ)……
求一棵樹的帶權(quán)重心,支持修改權(quán)值。
動(dòng)態(tài)樹分治,也叫點(diǎn)分樹。 就是把每層的重心連成一棵樹,然后在這棵樹上亂搞(具體網(wǎng)上教程多)。
不過第一次寫這題暴力碾過去了…..好像還挺快的….
先講暴力 假設(shè)上一次找到的重心在u,那么如果在某一點(diǎn)v增加了權(quán)值,那當(dāng)前的重心一定是在u到v的相反方向上,只要沿著相反方向找就行了。
具體怎么找…可以這么想: 當(dāng)前結(jié)點(diǎn)為x,y為與x相鄰的結(jié)點(diǎn),w[x]為x結(jié)點(diǎn)上的權(quán)值,cnt為總權(quán)值,那么如果cnt-w[y]< w[y]即cnt<2*w[y]時(shí),往y點(diǎn)移動(dòng),直到不能移動(dòng)為止。至于為什么……自己腦補(bǔ)一下
w[x]可以用dfs序加BIT維護(hù)。
#include <cstdio>#include <iostream>#include <string>#include <cstring>#define inf 1ll<<60#define N 100010using namespace std;typedef long long ll;int G[N],n,m,tt,rt,l[N],r[N],tc,lcA[N][25],dp[N];ll B[N<<1],Ans,A[N],tot,ds[N];struct edge{ int w,t,nx;}E[N<<1];struct lef{ int f,w,d;}T[N];inline char C(){ static char buf[100000],*p1=buf,*p2=buf; if(p1==p2){ p2=(p1=buf)+fread(buf,1,100000,stdin); if(p1==p2) return EOF; } return *p1++;}inline void reaD(int &x){ char Ch=C();x=0;int f=1; for(;Ch>'9'||Ch<'0';Ch=C())if(Ch=='-')f=-1; for(;Ch>='0'&&Ch<='9';x=x*10+Ch-'0',Ch=C());x*=f;}inline void reaD(ll &x){ char Ch=C();x=0;ll f=1; for(;Ch>'9'||Ch<'0';Ch=C())if(Ch=='-')f=-1; for(;Ch>='0'&&Ch<='9';x=x*10+Ch-'0',Ch=C());x*=f;}inline void InserT(int x,int y,int w){ E[++tt].t=y;E[tt].nx=G[x];E[tt].w=w;G[x]=tt; E[++tt].t=x;E[tt].nx=G[y];E[tt].w=w;G[y]=tt;}inline void build(int g,int f,int d){ T[g].f=f;l[g]=++tc;T[g].d=d; dp[g]=dp[f]+1;ds[g]=ds[f]+d; lcA[g][0]=f; for(int i=1;i<=20;i++) lcA[g][i]=lcA[lcA[g][i-1]][i-1]; for(int i=G[g];i;i=E[i].nx) if(E[i].t!=f) build(E[i].t,g,E[i].w); r[g]=++tc;}inline void add(int x,int y){ for(;x<=tc;x+=x&-x) B[x]+=y;}inline ll query(int x){ ll res=0; for(;x;x-=x&-x) res+=B[x]; return res;}inline ll Qlw(int x){ return query(r[x])-query(l[x]);}int w[30],wt;inline void Pt(ll x){ if(!x){putchar(48);putchar('/n');return;} if(x<0){putchar('-');x=-x;}; while(x)w[++wt]=x%10,x/=10; for(;wt;wt--)putchar(48+w[wt]);putchar('/n');}inline void swap(int &x,int &y){ int z=x;x=y;y=z;}inline int clca(int x,int y){ if(dp[x]<dp[y]) swap(x,y); int delt=dp[x]-dp[y],i; for(i=0;i<=20;i++) if(delt&(1<<i)) x=lcA[x][i]; while(x!=y){ for(i=-1;i;i++) if(lcA[x][i+1]==lcA[y][i+1]) break; if(i==-1) return lcA[x][0]; x=lcA[x][i];y=lcA[y][i]; } return x;}int main(){ freopen("tree.in","r",stdin); freopen("tree.out","w",stdout); reaD(n);reaD(m); for(int i=1,x,y,w;i<n;i++)reaD(x),reaD(y),reaD(w),InserT(x,y,w); /*int ok=0; for(int i=1;i<=n;i++) if(du[i]>2){ok=1;break;} if(!ok){linktime();return 0;}*/ build(1,0,0);reaD(rt); reaD(A[rt]);tot+=A[rt]; add(r[rt],A[rt]); Pt(Ans=0); for(int i=1,x,y,j,lca;i<m;i++){ reaD(x);reaD(y); add(r[x],y);A[x]+=y;tot+=y; lca=clca(x,rt);Ans+=1ll*(ds[x]+ds[rt]-2*ds[lca])*y; //S(rt,0,Ans); while(1){ if(T[rt].f&&tot>2ll*Qlw(rt)){Ans-=1ll*(tot-2*Qlw(rt))*T[rt].d;rt=T[rt].f;continue;} for(j=G[rt];j;j=E[j].nx) if(T[rt].f!=E[j].t&&tot<2ll*Qlw(E[j].t)){ Ans-=1ll*(2*Qlw(E[j].t)-tot)*E[j].w; rt=E[j].t; break; } if(!j) break; } Pt(Ans); } return 0;}復(fù)雜度是
正確做法是點(diǎn)分樹。 建出點(diǎn)分樹,每次找只要從根節(jié)點(diǎn)開始分治地找就行了。
復(fù)雜度
事實(shí)證明如果考場(chǎng)上想不出標(biāo)算或沒時(shí)間寫標(biāo)算,這種信仰暴力還是可以接受的……
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注