CF109 C. Lucky Tree 并查集
Petya loves lucky numbers. We all know that lucky numbers are the positive integers whose decimal representations contain only the lucky digits 4 and 7. For example, numbers 47, 744, 4 are lucky and 5, 17, 467 are not.
One day Petya encountered a tree with n vertexes. Besides, the tree was weighted, i. e. each edge of the tree has weight (a positive integer). An edge is lucky if its weight is a lucky number. Note that a tree with n vertexes is an undirected connected graph that has exactly n - 1 edges.
Petya wondered how many vertex triples (i, j, k) exists that on the way from i to j, as well as on the way from i to k there must be at least one lucky edge (all three vertexes are pairwise distinct). The order of numbers in the triple matters, that is, the triple (1, 2, 3) is not equal to the triple (2, 1, 3) and is not equal to the triple (1, 3, 2).
Find how many such triples of vertexes exist.
The first line contains the single integer n (1 ≤ n ≤ 105) — the number of tree vertexes. Next n - 1 lines contain three integers each: ui vi wi (1 ≤ ui, vi ≤ n, 1 ≤ wi ≤ 109) — the pair of vertexes connected by the edge and the edge's weight.
On the single line print the single number — the answer.
Please do not use the %lld specificator to read or write 64-bit numbers in С++. It is recommended to use the cin, cout streams or the %I64d specificator.
4
1 2 4
3 1 2
1 4 7
16
4
1 2 4
1 3 47
1 4 7447
24
The 16 triples of vertexes from the first sample are:(1, 2, 4), (1, 4, 2), (2, 1, 3), (2, 1, 4), (2, 3, 1), (2, 3, 4), (2, 4, 1), (2, 4, 3), (3, 2, 4), (3, 4, 2), (4, 1, 2), (4, 1, 3), (4, 2, 1), (4, 2, 3), (4, 3, 1), (4, 3, 2).
In the second sample all the triples should be counted: 4·3·2 = 24.
题意:
一条边,如果这一条边的权值只由数字4,7组成,我们就说这条边是一条幸运边,否则不是
现在有一棵树,边有权值,问:
这棵树有多少点对(i,j,k)满足:
i!=j!=k,
并且路径i到j至少经过一条幸运边,路径i到k也至少经过一条幸运边
注意:
点对(i,j,k)和(j,i,k)和(i,k,j)算是不同的点对
这道题,直接算满足的点对好像有点麻烦,正难则反,我们先算出不符合的点对
符合的点对=总点对-不符合的点对
总点对=n*(n-1)*(n-2)
现在考虑不符合的点对
不符合的点对分2种:
1.i到j和i到k2条路径都不包含幸运边
2.2条路径,其中一条经过了幸运边,另外一条没有经过幸运边
这棵树,根据幸运边我们可以分成若干个联通块,这里可以用并查集来实现
find_fa()函数的同时维护数组sum
sum[i]表示i所在的联通块的节点数
接着我们枚举联通块
对于每一个连通块:
2条路径都没有经过幸运边的数量:sum[i]*(sum[i]-1)*(sum[i]-2)
只有一条经过幸运边的数量:2*sum[i]*(sum[i]-1)*(n-sum[i])
累加就得到所有不符合的点对了
注意:
1.并查集记得路径压缩
2.每一个连通块只算一次,不要重复计算
代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream> #define LL long long using namespace std; const int maxn=1e5+; int fa[maxn];
LL sum[maxn]; void solve(); int main()
{
solve();
return ;
} //初始化
void init(int n)
{
for(int i=;i<=n;i++){
fa[i]=i;
sum[i]=;
}
} //判断是不是幸运边
bool ok(int w)
{
while(w){
int cur=w%;
if(cur!= && cur!=)
return false;
w/=;
}
return true;
} //并查集,注意要路径压缩,注意sum数组的更新
int find_fa(int x)
{
if(fa[x]==x)
return x;
int cur=find_fa(fa[x]);
sum[cur]+=sum[x];
sum[x]=;
fa[x]=cur;
return cur;
} void solve()
{
int n;
scanf("%d",&n);
init(n);
for(int i=;i<n;i++){
int u,v,w;
scanf("%d %d %d",&u,&v,&w);
if(!ok(w)){
int fau=find_fa(u);
int fav=find_fa(v);
if(fau!=fav){
fa[fau]=fav;
sum[fav]+=sum[fau];
sum[fau]=;
}
}
}
if(n<){
printf("0\n");
return ;
}
LL ans=;
for(int i=;i<=n;i++){
if(find_fa(i)==i){
if(sum[i]>)
ans+=(sum[i]*(sum[i]-)*(sum[i]-));
ans+=2LL*sum[i]*(sum[i]-)*(n-sum[i]);
}
}
ans=n*(n-1LL)*(n-2LL)-ans; printf("%I64d\n",ans);
return ;
}
CF109 C. Lucky Tree 并查集的更多相关文章
- Hdu.1325.Is It A Tree?(并查集)
		Is It A Tree? Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) To ... 
- Is It A Tree?(并查集)
		Is It A Tree? Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 26002 Accepted: 8879 De ... 
- HDU 5606 tree 并查集
		tree 把每条边权是1的边断开,发现每个点离他最近的点个数就是他所在的连通块大小. 开一个并查集,每次读到边权是0的边就合并.最后Ansi=size[findset(i)],size表示每个并 ... 
- [Swust OJ 856]--Huge Tree(并查集)
		题目链接:http://acm.swust.edu.cn/problem/856/ Time limit(ms): 1000 Memory limit(kb): 10000 Description T ... 
- Codeforces Round #363 (Div. 2)D. Fix a Tree(并查集)
		D. Fix a Tree time limit per test 2 seconds memory limit per test 256 megabytes input standard input ... 
- Is It A Tree?(并查集)(dfs也可以解决)
		Is It A Tree? Time Limit:1000MS Memory Limit:10000KB 64bit IO Format:%I64d & %I64u Submi ... 
- tree(并查集)
		tree Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total Submis ... 
- 树上统计treecnt(dsu on tree 并查集 正难则反)
		题目链接 dalao们怎么都写的线段树合并啊.. dsu跑的好慢. \(Description\) 给定一棵\(n(n\leq 10^5)\)个点的树. 定义\(Tree[L,R]\)表示为了使得\( ... 
- hdu 1325 Is It A Tree? 并查集
		Is It A Tree? Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Tot ... 
随机推荐
- Codeforces Round #122 (Div. 2)
			A. Exams 枚举分数为3.4.5的数量,然后计算出2的数量即可. B. Square 相当于求\(\min{x(n+1)\ \%\ 4n=0}\) 打表发现,对\(n\ \%\ 4\)分类讨论即 ... 
- 1-3-2 Windows应用程序常用消息
			主要内容:介绍Windows编程中常用的消息 1.WM_LBUTTONDOWN产生单击鼠标左键的消息 lParam: 低字节包含当前光标的X坐标值 X = LOWORD(lParam); 高字节包含当 ... 
- HDU 2717 Catch That Cow --- BFS
			HDU 2717 题目大意:在x坐标上,农夫在n,牛在k.农夫每次可以移动到n-1, n+1, n*2的点.求最少到达k的步数. 思路:从起点开始,分别按x-1,x+1,2*x三个方向进行BFS,最先 ... 
- springMvc源码学习之:spirngMVC获取请求参数的方法2
			@RequestParam,你一定见过:@PathVariable,你肯定也知道:@QueryParam,你怎么会不晓得?!还有你熟悉的他 (@CookieValue)!她(@ModelAndView ... 
- Linux-Nginx之sendfile与上下文切换
			今天在看nginx thread pool的时候,频繁的看到sendfile,其实以前也经常看到sendfile,只是我平时选择性的忽视而已... 先说下sendfile,明天在好好聊下nginx 线 ... 
- 20150914 异常语句 math的方法 去空格 索引
			异常语句 int cuo = 0; Console.WriteLine("请输入:"); string s = Console.ReadLine(); try { DateTime ... 
- ABBYY导出结果为PDF注意事项
			使用ABBYY FineReader Pro for Mac OCR文字识别软件识别文档时,可以将已识别的文本保存到文件中,还可以通过电子邮件发送已识别的文本,只要输出格式受FineReader支持. ... 
- JDBC - Oracle PreparedStatement (GeneratedKey kind) ArrayIndexOutOfBoundsException
			问题: Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 12at oracle.jdbc. ... 
- RMAN备份与恢复之DataBase
			1 准备 [oracle@TEST144239 /]$ sqlplus /nolog SQL Production :: Copyright (c) , , Oracle. All rights ... 
- GitHub简单使用入门
			自从google code关闭了下载服务了之后,GitHub作为了目前最好用的免费开源项目托管站点,众多开源项目都托管在github,其中不乏著名的播放器MPC-HC. 不习惯于英文的朋友,难免少不了 ... 
