题目链接

Problem Description

There is a tree with n nodes, each of which has a type of color represented by an integer, where the color of node i is ci.

The path between each two different nodes is unique, of which we define the value as the number of different colors appearing in it.

Calculate the sum of values of all paths on the tree that has n(n−1)2 paths in total.

Input

The input contains multiple test cases.

For each test case, the first line contains one positive integers n, indicating the number of node. (2≤n≤200000)

Next line contains n integers where the i-th integer represents ci, the color of node i. (1≤ci≤n)

Each of the next n−1 lines contains two positive integers x,y (1≤x,y≤n,x≠y), meaning an edge between node x and node y.

It is guaranteed that these edges form a tree.

Output

For each test case, output "Case #x: y" in one line (without quotes), where x indicates the case number starting from 1 and y denotes the answer of corresponding case.

Sample Input

3

1 2 1

1 2

2 3

6

1 2 1 3 2 1

1 2

1 3

2 4

2 5

3 6

Sample Output

Case #1: 6

Case #2: 29

分析:

给你一棵树 n个点,n个点有自己的颜色,每两个点的权值就是这条路上不同的颜色的种类数,问整棵树的权值。。。首先可以想到的。。一棵树上所有的边是 n×(n-1)/2 假如每个颜色都在所有的边上,那么总权值就是 总颜色 num×n×(n-1)/2 那么很简单的一个想法 是

总权值 减去多加上的权值,那么多加上的权值就是每个点的颜色不在的边的数量的总和,由于时间要求,你可能需要在on 内得到这些东西,所以你可以用类似树形dp维护,那么如何得到这个值呢。画个树的图,可以想到,每个树节点下面支路都是相连的,每一部分的求和都可以用 n×(n-1)/2。然后再运用虚树的思想,每个点管理下面它需要管理的部分。。(即只有它自身的这种颜色,别的颜色看作无色,那么每个节点的管理部分都是他下面的点),你可能会想如果每次都处理的是自己下面的点,那么隔壁树枝的怎么办呢,由于这是一个分治合并的过程,在树上那么总会有一个汇点处理这一切的。。。

那么现在要找的路径数 就是 把这个颜色的点都去掉,得到了很多的块或者点,每个块内部的路径可以用n×(n-1)/2 得到,然而每个部分都有相应管辖的点.

#include <bits/stdc++.h>
using namespace std;
const int N = 4e5+100;
typedef long long ll;
int c[N],vis[N];///c[i]表示第i个点的颜色,vis[i]表示的是i这个颜色有没有出现过
vector<int> e[N];///存储这个图呢
ll sum[N],size[N];///sum[i]表示的是这个颜色i所能管理的点,size[i]表示的是以i为根的点的个数
ll ans;
void dfs(int x,int y)
{ size[x]=1;/// 指的是x为根的树的点的个数
sum[c[x]]++;/// sum是指 当前颜色的所能管理的点(包括自身的总个数) 那么每种颜色 最后都是一个接近根的值
///到不了根的部分 就要在后面剔除
ll pre=sum[c[x]];/// 指从根到 x ,c[x] 管辖的点的总个数
for(int i=0; i<e[x].size(); i++)
{
if(e[x][i]==y) continue;
dfs(e[x][i],x);
size[x]+=size[e[x][i]];/// 以x为根的树的点的总个数,当前的这个点还要加上他的子数上的点
ll count=size[e[x][i]]-(sum[c[x]]-pre);/// 总的减去被管理的部分 剩下的都是自由的联通块 直接求路径后面剔除
ans=ans+(1LL*count*(count-1))/2;
//printf("%d\n",sum[c[x]]);
sum[c[x]]+=count;/// 加上这些被剔除的,相当于把这些与根节点颜色不一样的点标记成颜色一样的,这样递归往上返回的时候就不会重复计算这些点了
pre=sum[c[x]];///更新这个颜色所能管理的所有的点
}
} int main()
{
int n,cas=1;
while(scanf("%d",&n)!=EOF)
{
int num=0;
ans=0;
memset(sum,0,sizeof(sum));
memset(vis,0,sizeof(vis));
for(int i=1; i<=n; i++)
{
e[i].clear();
scanf("%d",&c[i]);
if(!vis[c[i]])
{
vis[c[i]]=1;///标记这个颜色有没有出现过
num++;///不同的颜色数
}
}
for(int i=1; i<n; i++)
{
int u,v;
scanf("%d%d",&u,&v);
e[u].push_back(v);
e[v].push_back(u);
}
dfs(1,0);
ll ANS = 1LL*num*((1LL)*n*(n-1))/2;
/// 因为根只有一种颜色 这里要剔除所有到不了根的颜色
for(int i=1; i<=n; i++)
{
if(vis[i])
{
ll ct=n-sum[i];
ans+=ct*(ct-1)/2;
}
}
printf("Case #%d: %lld\n", cas++, ANS-ans);
}
}

2017ACM暑期多校联合训练 - Team 1 1003 HDU 6035 Colorful Tree (dfs)的更多相关文章

  1. 2017ACM暑期多校联合训练 - Team 6 1003 HDU 6098 Inversion (模拟)

    题目链接 Problem Description Give an array A, the index starts from 1. Now we want to know Bi=maxi∤jAj , ...

  2. 2017ACM暑期多校联合训练 - Team 4 1003 HDU 6069 Counting Divisors (区间素数筛选+因子数)

    题目链接 Problem Description In mathematics, the function d(n) denotes the number of divisors of positiv ...

  3. 2017ACM暑期多校联合训练 - Team 3 1003 HDU 6058 Kanade's sum (模拟)

    题目链接 Problem Description Give you an array A[1..n]of length n. Let f(l,r,k) be the k-th largest elem ...

  4. 2017ACM暑期多校联合训练 - Team 2 1003 HDU 6047 Maximum Sequence (线段树)

    题目链接 Problem Description Steph is extremely obsessed with "sequence problems" that are usu ...

  5. 2017ACM暑期多校联合训练 - Team 4 1004 HDU 6070 Dirt Ratio (线段树)

    题目链接 Problem Description In ACM/ICPC contest, the ''Dirt Ratio'' of a team is calculated in the foll ...

  6. 2017ACM暑期多校联合训练 - Team 9 1005 HDU 6165 FFF at Valentine (dfs)

    题目链接 Problem Description At Valentine's eve, Shylock and Lucar were enjoying their time as any other ...

  7. 2017ACM暑期多校联合训练 - Team 9 1010 HDU 6170 Two strings (dp)

    题目链接 Problem Description Giving two strings and you should judge if they are matched. The first stri ...

  8. 2017ACM暑期多校联合训练 - Team 8 1006 HDU 6138 Fleet of the Eternal Throne (字符串处理 AC自动机)

    题目链接 Problem Description The Eternal Fleet was built many centuries ago before the time of Valkorion ...

  9. 2017ACM暑期多校联合训练 - Team 8 1002 HDU 6134 Battlestation Operational (数论 莫比乌斯反演)

    题目链接 Problem Description The Death Star, known officially as the DS-1 Orbital Battle Station, also k ...

随机推荐

  1. 软工网络15团队作业4-DAY2

    每个人的工作 (有work item 的ID),并将其记录在码云项目管理中: 昨天已完成的工作. 张陈东芳:查看数据库的连接 吴敏烽:规范商品实体类 周汉麟:研究获取商品信息的方法 林振斌:研究获取商 ...

  2. PHP面向对象之重写

    覆盖(override): 基本概念 覆盖,又叫“重写”: 含义: 将一个类从父类中继承过来的属性和方法“重新定义”——此时相当于子类不想用父类的该属性或方法,而是想要定义. 覆盖的现实需要: 对于一 ...

  3. react-router之代码分离

    概念 无需用户下载整个应用之后才能访问访问它.即边访问边下载.因此我们设计一个组件<Bundle>当用户导航到它是来动态加载组件. import loadSomething from 'b ...

  4. day 008 文件操作

    08. 万恶之源-⽂文件操作本节主要内容:1. 初识⽂文件操作2. 只读(r, rb)3. 只写(w, wb)4. 追加(a, ab)5. r+读写6. w+写读7. a+写读(追加写读)8. 其他操 ...

  5. 【明哥报错簿】之json转换报错---net.sf.ezmorph.bean.MorphDynaBean cannot be cast to XXXDO

    简单的json和bean转换直接用: public static void main(String[] args) { String s = "{'request': [{'orderCod ...

  6. (转)slf4j+logback将日志输出到控制台

    因为博主不允许转载...这边做链接记录 http://blog.csdn.net/gsycwh/article/details/52972946

  7. 【BZOJ3894】文理分科(最小割)

    [BZOJ3894]文理分科(最小割) 题面 BZOJ Description 文理分科是一件很纠结的事情!(虽然看到这个题目的人肯定都没有纠 结过) 小P所在的班级要进行文理分科.他的班级可以用一个 ...

  8. 测试开发linux面试之三:后台进程之操作

    Hi,大家好我是Tom,继上次分享之后这次给大家带来新的知识. 进程是Linux系统中一个非常重要的概念.Linux是一个多任务的操作系统,系统上经常同时运行着多个进程.我们不关心这些进程究竟是如何分 ...

  9. BZOJ3509 [CodeChef] COUNTARI 【分块 + fft】

    题目链接 BZOJ3509 题解 化一下式子,就是 \[2A[j] = A[i] + A[k]\] 所以我们对一个位置两边的数构成的生成函数相乘即可 但是由于这样做是\(O(n^2logn)\)的,我 ...

  10. BZOJ3110 K大数查询 【线段树 + 整体二分 或 树套树(非正解)】

    Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个位置到第b个位 ...