• 【BZOJ2152】聪聪可可(点分治)


    【BZOJ2152】聪聪可可(点分治)

    题面

    Description

    聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃、两个人都想玩儿电脑(可是他们家只有一台电脑)……遇到这种问题,一般情况下石头剪刀布就好了,可是他们已经玩儿腻了这种低智商的游戏。他们的爸爸快被他们的争吵烦死了,所以他发明了一个新游戏:
    由爸爸在纸上画 n 个“点”,并用 n−1 条“边”把这 n个“点”恰好连通(其实这就是一棵树)。并且每条“边”上都有一个数。接下来由聪聪和可可分别随即选一个点(当然他们选点时是看不到这棵树的),如果两个点之间所有边上数的和加起来恰好是3的倍数,则判聪聪赢,否则可可赢。
    聪聪非常爱思考问题,在每次游戏后都会仔细研究这棵树,希望知道对于这张图自己的获胜概率是多少。现请你帮忙求出这个值以验证聪聪的答案是否正确。

    Input

    输入的第1行包含1个正整数 n。
    后面n-1行,每行3个整数 x、y、w,表示 x 号点和 y 号点之间有一条边,上面的数是 w。

    Output

    以即约分数形式输出这个概率(即“ a/b ”的形式,其中 a 和 b 必须互质。如果概率为 1,输出“ 1/1 ”)。

    Sample Input

    5
    1 2 1
    1 3 2
    1 4 1
    2 5 3

    Sample Output

    13/25

    Hint

    样例说明:
    13组点对分别是(1,1) (2,2) (2,3) (2,5) (3,2) (3,3) (3,4) (3,5) (4,3) (4,4) (5,2) (5,3) (5,5)。
    数据规模和约定
    对于30%的数据,n≤1000  另有20%的数据,给出的树中每个节点的度不超过2;
    对于100%的数据,n≤20000

    题解

    还是点分治
    因为求点对的距离关于3的膜
    因此只要统计子树中的距离膜3的点个数即可

    #include<iostream>
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<cmath>
    #include<algorithm>
    #include<set>
    #include<map>
    #include<vector>
    #include<queue>
    using namespace std;
    #define MAX 30000
    inline int read()
    {
        int x=0,t=1;char ch=getchar();
        while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
        if(ch=='-')t=-1,ch=getchar();
        while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
        return x*t;
    }
    struct Line
    {
    	int v,next,w;
    }e[MAX<<1];
    int h[MAX],cnt=1;
    int size[MAX],Size,minr,root;
    int S[3],tot,num[3];
    bool vis[MAX];
    int n;
    inline void Add(int u,int v,int w)
    {
    	e[cnt]=(Line){v,h[u],w};
    	h[u]=cnt++;
    }
    void Getroot(int u,int ff)
    {
    	size[u]=1;
    	int ret=0;
    	for(int i=h[u];i;i=e[i].next)
    	{
    		int v=e[i].v;
    		if(v==ff||vis[v])continue;
    		Getroot(v,u);
    		size[u]+=size[v];
    		ret=max(ret,size[v]);
    	}
    	ret=max(ret,Size-size[u]);
    	if(ret<minr)minr=ret,root=u;
    }
    void Getdep(int u,int ff,int dd)
    {
    	S[dd%3]++;
    	for(int i=h[u];i;i=e[i].next)
    	{
    		int v=e[i].v;
    		if(v==ff||vis[v])continue;
    		Getdep(v,u,(dd+e[i].w)%3);
    	}
    }
    void Calc(int u,int fl,int pr)
    {
    	memset(S,0,sizeof(S));
    	Getdep(u,u,0);
    	if(fl)
    	{
    		num[0]+=2*S[1]*S[2]+S[0]*S[0];
    		num[1]+=2*S[0]*S[1]+S[2]*S[2];
    		num[2]+=2*S[0]*S[2]+S[1]*S[1];
    	}
    	else
    	{
    		pr%=3;
    		num[(0+pr)%3]-=2*S[1]*S[2]+S[0]*S[0];
    		num[(1+pr)%3]-=2*S[0]*S[1]+S[2]*S[2];
    		num[(2+pr)%3]-=2*S[0]*S[2]+S[1]*S[1];
    	}
    	
    }
    void DFS(int u)
    {
    	Calc(u,1,0);
    	vis[u]=true;
    	for(int i=h[u];i;i=e[i].next)
    	{
    		int v=e[i].v;
    		if(vis[v])continue;
    		Calc(v,0,e[i].w*2);
    		minr=n;Size=size[v];
    		Getroot(v,u);
    		DFS(root);
    	}
    }
    int gcd(int a,int b)
    {
    	return !a?b:gcd(b%a,a);
    }
    int main()
    {
    	Size=n=read();
    	for(int i=1,u,v,w;i<n;++i)
    	{
    		u=read(),v=read(),w=read();
    		Add(u,v,w);Add(v,u,w);
    	}
    	minr=n;Getroot(1,1);
    	DFS(root);
    	int tt=num[0]+num[1]+num[2];
    	int dd=gcd(tt,num[0]);
    	printf("%d/%d
    ",num[0]/dd,tt/dd);
    	return 0;
    }
    
  • 相关阅读:
    H5测试
    mysql安装
    HTTP工作过程
    正则表达式常用方法
    20223
    视音频数据处理入门:RGB、YUV像素数据处理
    父母课堂——如何陪伴孩子过一个有意义的寒假
    c语言二维数组
    MobaXterm
    蓝牙介绍
  • 原文地址:https://www.cnblogs.com/cjyyb/p/8010518.html
Copyright © 2020-2023  润新知