BZOJ2152聪聪可可
bzoj传送门
luogu传送门
这题算是很sb的点分治了,最近在点分治复习,写了练练手,对于这个题只需要对统计0,1,2出现的次数就好了吧,然后发现答案不对,也就是每个点对需要算两遍嘛,0也算,所以答案再加个n就好啦
(用树状数组的\(O(nlog^2)\))代码:
#include<cstdio>
#define max(a,b) (a>b?a:b)
int ans,rt,sum,w,size[20001],f[4],son[20001],dis[20001],n,pre[40001],nxt[40001],h[20001],v[40001],cnt;
bool vis[20001];
void add(int x,int y,int z)
{
pre[++cnt]=y;nxt[cnt]=h[x];h[x]=cnt;v[cnt]=z;
pre[++cnt]=x;nxt[cnt]=h[y];h[y]=cnt;v[cnt]=z;
}
void getroot(int x,int fa)
{
size[x]=1;son[x]=0;
for(int i=h[x];i;i=nxt[i])
if(pre[i]!=fa&&!vis[pre[i]])
{
getroot(pre[i],x);
size[x]+=size[pre[i]];
son[x]=max(size[pre[i]],son[x]);
}
son[x]=max(son[x],sum-size[x]);
if(son[x]<son[rt])rt=x;
}
void getans(int x,int fa)
{
if(dis[x]!=0)ans+=f[3-dis[x]];
else ans+=f[0]+1;
for(int i=h[x];i;i=nxt[i])
if(!vis[pre[i]]&&pre[i]!=fa)
{
dis[pre[i]]=(dis[x]+v[i])%3;
getans(pre[i],x);
}
}
void getdeep(int x,int fa)
{
f[dis[x]]++;
for(int i=h[x];i;i=nxt[i])if(!vis[pre[i]]&&pre[i]!=fa)getdeep(pre[i],x);
}
void dfs(int x)
{
vis[x]=1;
for(int i=h[x];i;i=nxt[i])
if(!vis[pre[i]])
{
dis[pre[i]]=v[i]%3;
getans(pre[i],0),getdeep(pre[i],0);
}
f[0]=f[1]=f[2]=0;
for(int i=h[x];i;i=nxt[i])
if(!vis[pre[i]])
{
rt=0;sum=size[pre[i]];getroot(pre[i],0);
dfs(rt);
}
}
int gcd(int a,int b)
{
if(!b)return a;
return gcd(b,a%b);
}
int main()
{
scanf("%d",&n);
for(int i=1,x,y,z;i<n;i++)scanf("%d%d%d",&x,&y,&z),add(x,y,z);
sum=son[0]=n;getroot(1,0);dfs(rt);
ans=ans*2+n;w=n*n;
printf("%d/%d\n",ans/gcd(ans,w),w/gcd(ans,w));
}
(当年写的two pointer(也称尺取法)):
#include<cstdio>
#include<algorithm>
using namespace std;
int now,st[20001],n,cnt,sum,ans,pre[40001],rt,nxt[40001],v[40001],h[20001],son[20001],size[20001],dis[20001],t[3];
bool vis[20001];
void read(int &x)
{
int f=1;x=0;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-1;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
x*=f;
}
void add(int x,int y,int z)
{
pre[++cnt]=y;nxt[cnt]=h[x];h[x]=cnt,v[cnt]=z;
pre[++cnt]=x;nxt[cnt]=h[y];h[y]=cnt,v[cnt]=z;
}
void getroot(int x,int fa)
{
size[x]=1;son[x]=0;
for(int i=h[x];i;i=nxt[i])
if(!vis[pre[i]]&&pre[i]!=fa)
{
getroot(pre[i],x);
size[x]+=size[pre[i]];
son[x]=max(son[x],size[pre[i]]);
}
son[x]=max(son[x],sum-size[x]);
if(son[x]<son[rt])rt=x;
}
void get(int x,int fa)
{
t[dis[x]%3]++;
for(int i=h[x];i;i=nxt[i])
if(!vis[pre[i]]&&pre[i]!=fa)
{
dis[pre[i]]=dis[x]+v[i];
get(pre[i],x);
}
}
int getans(int x)
{
now=0;
get(x,0);
return t[0]*t[0]+t[1]*t[2]*2;
}
void dfs(int x)
{
vis[x]=1;dis[x]=0;
ans+=getans(x);
t[0]=t[1]=t[2]=0;
for(int i=h[x];i;i=nxt[i])
if(!vis[pre[i]])
{
dis[pre[i]]=v[i];
ans-=getans(pre[i]);
t[0]=t[1]=t[2]=0;
rt=0;sum=size[pre[i]];
getroot(pre[i],0);
dfs(rt);
}
}
int gcd(int a,int b){return !b?a:gcd(b,a%b);}
int main()
{
read(n);
for(int i=1,x,y,z;i<n;i++)
read(x),read(y),read(z),add(x,y,z);
sum=n;son[0]=n;
getroot(1,0);
dfs(rt);
int mod=gcd(ans,n*n);
printf("%d/%d",ans/mod,n*n/mod);
}
BZOJ2152聪聪可可的更多相关文章
- 【BZOJ2152】聪聪可可(点分治)
[BZOJ2152]聪聪可可(点分治) 题面 Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电 ...
- BZOJ2152 [国家集训队] 聪聪可可 [点分治]
题目传送门 聪聪可可 Time Limit: 3 Sec Memory Limit: 259 MBSubmit: 5237 Solved: 2750[Submit][Status][Discuss ...
- BZOJ2152 聪聪可可 【点分治】
BZOJ2152 聪聪可可 Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电脑)--遇到这种问 ...
- 【bzoj2152】聪聪可可 点分治
[bzoj2152]聪聪可可 2014年9月7日3,5472 Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是 ...
- BZOJ2152 聪聪可可 (点分治)
2152: 聪聪可可 题意: 在一棵边带权的树中,问任取两个点,这两个点间的权值和是3的倍数的概率. 思路: 经典的点分治题目. 利用点分治在计算所有路径长度,把路径长度对3取模,用t[0],t[1] ...
- [bzoj2152][聪聪和可可] (点分治+概率)
Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电脑)……遇到这种问题,一般情况下石头剪刀布就好 ...
- C++之路进阶——bzoj2152(聪聪可可)
F.A.Qs Home Discuss ProblemSet Status Ranklist Contest ModifyUser hyxzc Logout 捐赠本站 Notice:由于本OJ建立在 ...
- bzoj2152 聪聪可可
Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电脑)……遇到这种问题,一般情况下石头剪刀布就好 ...
- 【BZOJ2152】聪聪可可
Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电脑)……遇到这种问题,一般情况下石头剪刀布就好 ...
- BZOJ2152[国家集训队]聪聪可可——点分治
题目描述 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电脑)……遇到这种问题,一般情况下石头剪刀布就好了,可是他们已 ...
随机推荐
- MapReduce源代码分析之LocatedFileStatusFetcher
LocatedFileStatusFetcher是MapReduce中一个针对给定输入路径数组,使用配置的线程数目来获取数据块位置的有用类. 它的主要作用就是利用多线程技术.每一个线程相应一个任务.每 ...
- Axure实现Tab选项卡切换功能
这几天用Axure画原型图的过程中,须要实现Tab选项卡切换的效果,但Axure中并没有类似于Tab控件的部件,所以能够用Axure中的动态面板(Dynamic Panel)来实现. 本文以已经汉化的 ...
- 20170316 REUSE_alv_display_lvc 面向对象函数
**将ALV显示数据更新进输出内表中 DATA: LR_GRID TYPE REF TO CL_GUI_ALV_GRID. CALL FUNCTION 'GET_GLOBALS_FROM_SLV ...
- appium():PageObject&PageFactory
Appium Java client has facilities which components to Page Object design pattern and Selenium PageFa ...
- RequireJS 配置理解
RequireJS 配置: 1.首先加载RequireJS文件 <script src="//cdn.bootcss.com/require.js/2.1.22/require.js& ...
- flume 日志收集单节点
flume 是 cloudera公司研发的日志收集系统,采用3层结构:1. agent层,用于直接收集日志;2.connect 层,用于接受日志; 3. 数据存储层,用于保存日志.由一到多个maste ...
- coeforces 665D D. Simple Subset(最大团orsb题)
题目链接: D. Simple Subset time limit per test 1 second memory limit per test 256 megabytes input standa ...
- 51nod1060:最复杂的数(DFS求反素数)
把一个数的约数个数定义为该数的复杂程度,给出一个n,求1-n中复杂程度最高的那个数. 例如:12的约数为:1 2 3 4 6 12,共6个数,所以12的复杂程度是6.如果有多个数复杂度相等,输出最 ...
- 如何应用AutoIt,把局域网中所有的机器名展示在一个combox中?
有时候,我们会遇到以下情况: 你想与局域网中的某台机器建立连接,你就需要输入对方的机器名. 现在我比较懒,我不想输入对方的机器名,或者对方的机器名很难记住,那怎么办呢? 那就做一个combox在页面上 ...
- $bzoj4569$
$st表+并查集$ $考虑暴力方法:我们每次将对应相等的位置用并查集连起来,那么最终答案就是9*10^{连通块个数-1}$ $很明显上面这个办法过不去,问题在于重复次数太多了,如果一个区间已经对应相等 ...