zoukankan      html  css  js  c++  java
  • P2634 [国家集训队]聪聪可可 点分治

    思路:点分治

    提交:1次

    题解:

    不需要什么容斥。。。接着板子题说:
    还是基本思路:对于一颗子树,与之前的子树做贡献。
    我们把路径的权值在(\%3)意义下分类,即开三个桶(c[0],c[1],c[2]),分别记录每一类的路径条数。合并的时候显然有:

    [c[0]cdot mem[0]cdot 2+c[1]cdot mem[2]cdot 2+c[2]cdot mem[1]cdot 2 ]

    其中(mem[0],mem[1],mem[2])代表已经遍历的子树中每一类的路径条数。

    代码:

    #include<bits/stdc++.h>
    #define R register int
    using namespace std;
    namespace Luitaryi {
    template<class I> inline I g(I& x) { x=0; register I f=1;
    	register char ch; while(!isdigit(ch=getchar())) f=ch=='-'?-1:f;
    	do x=x*10+(ch^48); while(isdigit(ch=getchar())); return x*=f;
    } const int N=20010,Inf=1e+9;
    int n,m,cnt,rt,sum,ans; bool vis[N];
    int vr[N<<1],nxt[N<<1],w[N<<1],fir[N],d[N],sz[N],c[3],mem[3],mx[N];
    inline void add(int u,int v,int ww) {
    	vr[++cnt]=v,nxt[cnt]=fir[u],w[cnt]=ww,fir[u]=cnt;
    	vr[++cnt]=u,nxt[cnt]=fir[v],w[cnt]=ww,fir[v]=cnt;
    }
    inline void getsz(int u,int fa) {
    	sz[u]=1,mx[u]=0;
    	for(R i=fir[u];i;i=nxt[i]) { R v=vr[i];
    		if(v==fa||vis[v]) continue;
    		getsz(v,u); sz[u]+=sz[v];
    		mx[u]=max(mx[u],sz[v]);
    	} mx[u]=max(mx[u],sum-sz[u]);
    	if(mx[u]<mx[rt]) rt=u;
    }
    inline void getdis(int u,int fa) {
    	++c[d[u]]; for(R i=fir[u];i;i=nxt[i]) { R v=vr[i];
    		if(v==fa||vis[v]) continue;
    		d[v]=(d[u]+w[i])%3; getdis(v,u);
    	}
    }
    inline int calc() {
    	return c[0]*mem[0]*2+c[1]*mem[2]*2+c[2]*mem[1]*2;
    }
    inline void solve(int u,int fa) { vis[u]=true; mem[0]=1;
    	for(R i=fir[u];i;i=nxt[i]) { R v=vr[i];
    		if(v==fa||vis[v])	continue;
    		d[v]=w[i]%3; getdis(v,u);
    		ans+=calc(),mem[0]+=c[0],mem[1]+=c[1],mem[2]+=c[2];
    		memset(c,0,sizeof(c));
    	} memset(mem,0,sizeof(mem));
    	for(R i=fir[u];i;i=nxt[i]) { R v=vr[i];
    		if(v==fa||vis[v]) continue;
    		sum=sz[v],rt=0,mx[rt]=Inf;
    		getsz(v,u),getsz(rt,-1),solve(rt,u);
    	}
    }
    inline void main() {
    	g(n); for(R i=1,u,v,w;i<n;++i) g(u),g(v),g(w),add(u,v,w);
    	sum=n,mx[rt]=Inf; getsz(1,-1),getsz(rt,-1),solve(rt,-1); ans+=n;
    	R tmp=__gcd(ans,n*n); printf("%d/%d
    ",ans/tmp,n*n/tmp);
    }
    } signed main() {Luitaryi::main(); return 0;}
    

    2019.08.31
    69

  • 相关阅读:
    C#中的代理(Delegate)
    动态栈C语言
    AMS算法
    动态队列实现C语言
    带头结点的循环单链表C语言
    静态栈C语言
    不带头结点的单链表C语言实现
    带头结点的双向循环链表C语言
    带头节点的单链表C语言实现
    使用函数指针模拟C++多态
  • 原文地址:https://www.cnblogs.com/Jackpei/p/11437893.html
Copyright © 2011-2022 走看看