题目的意思是:给定一个点带颜色的树,两点之间的距离定义为路径上不同颜色的个数。求所有点对间的距离和。

做法有点分治,还有传说中的虚树DP,树上差分。

点分治法:

  考虑每个点的贡献,可以发现一个点的子树大小就是这个点的贡献。那么,对于同一个根的另一个子树的一个点x,去掉x到根结点对应颜色的贡献,再加上x到根结点上的颜色的种类数目,就是这个x点的答案。我们具体做的时候,是先不考虑根结点的,根结点对x点的贡献单独算。

  

#include <algorithm>
#include <iterator>
#include <iostream>
#include <cstring>
#include <cstdlib>
#include <iomanip>
#include <bitset>
#include <cctype>
#include <cstdio>
#include <string>
#include <vector>
#include <stack>
#include <cmath>
#include <queue>
#include <list>
#include <map>
#include <set>
#include <cassert> /* ⊂_ヽ
  \\ Λ_Λ 来了老弟
   \('ㅅ')
    > ⌒ヽ
   /   へ\
   /  / \\
   レ ノ   ヽ_つ
  / /
  / /|
 ( (ヽ
 | |、\
 | 丿 \ ⌒)
 | |  ) /
'ノ )  Lノ */ using namespace std;
#define lson (l , mid , rt << 1)
#define rson (mid + 1 , r , rt << 1 | 1)
#define debug(x) cerr << #x << " = " << x << "\n";
#define pb push_back
#define pq priority_queue typedef long long ll;
typedef unsigned long long ull;
//typedef __int128 bll;
typedef pair<ll ,ll > pll;
typedef pair<int ,int > pii;
typedef pair<int,pii> p3; //priority_queue<int> q;//这是一个大根堆q
//priority_queue<int,vector<int>,greater<int> >q;//这是一个小根堆q
#define fi first
#define se second
//#define endl '\n' #define boost ios::sync_with_stdio(false);cin.tie(0)
#define rep(a, b, c) for(int a = (b); a <= (c); ++ a)
#define max3(a,b,c) max(max(a,b), c);
#define min3(a,b,c) min(min(a,b), c); const ll oo = 1ll<<;
const ll mos = 0x7FFFFFFF; //
const ll nmos = 0x80000000; //-2147483648
const int inf = 0x3f3f3f3f;
const ll inff = 0x3f3f3f3f3f3f3f3f; //
const int mod = 1e9+;
const double esp = 1e-;
const double PI=acos(-1.0);
const double PHI=0.61803399; //黄金分割点
const double tPHI=0.38196601; template<typename T>
inline T read(T&x){
x=;int f=;char ch=getchar();
while (ch<''||ch>'') f|=(ch=='-'),ch=getchar();
while (ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x=f?-x:x;
} inline void cmax(int &x,int y){if(x<y)x=y;}
inline void cmax(ll &x,ll y){if(x<y)x=y;}
inline void cmin(int &x,int y){if(x>y)x=y;}
inline void cmin(ll &x,ll y){if(x>y)x=y;} /*-----------------------showtime----------------------*/
const int maxn = 2e5+;
int col[maxn];
vector<int>mp[maxn];
ll ans = , sumcol = ;
int sz[maxn],wt[maxn], root, curn;
int vis[maxn];
void findRoot(int u, int fa) {
sz[u] = ;wt[u] = ;
for(int i=; i<mp[u].size(); i++) {
int v = mp[u][i];
if(v == fa || vis[v]) continue;
findRoot(v, u);
sz[u] += sz[v];
wt[u] = max(sz[v], wt[u]);
}
wt[u] = max(wt[u], curn - sz[u]);
if(wt[u] <= wt[root]) root = u;
}
// map<int, int> pp;
ll pp[maxn];
int youmeiyou[maxn];
int ss;
void gao(int u, int fa, vector<pii>& vv, int cnt, ll sumfa, ll sum) {
ll res = ;
if(youmeiyou[col[u]] == )
vv.pb(pii(col[u], sz[u])), cnt++, res += pp[col[u]]; youmeiyou[col[u]]++;
ans += sumcol - sumfa - res + 1ll * cnt * sum;
//sum-color[根的颜色]+size[root]
if(youmeiyou[col[ss]] == ) ans += sum - pp[col[ss]];
for(int i=; i<mp[u].size(); i++) {
int v = mp[u][i];
if(fa == v || vis[v]) continue;
gao(v, u, vv, cnt, sumfa + res, sum);
}
youmeiyou[col[u]] --;
} void solve(int u) {
vis[u] = ;
findRoot(u, -);
ll sum = ;
sumcol = ;
queue<int>needclear;
needclear.push(col[u]);
for(int i=; i<mp[u].size(); i++) {
int v = mp[u][i];
if(vis[v]) continue;
vector<pii>vv;
ss = u;
gao(v, -, vv, , , sum); for(int j=; j<vv.size(); j++){
int c = vv[j].fi;
if(pp[c])pp[c] += vv[j].se;
else {
pp[c] = vv[j].se;
needclear.push(c);
}
sumcol += vv[j].se;
}
sum += sz[v];
} while(!needclear.empty()) {
pp[needclear.front()] = ;
needclear.pop();
}
for(int i=; i<mp[u].size(); i++) {
int v = mp[u][i];
if(!vis[v]) {
root = ; wt[] = inf; curn = sz[v];
findRoot(v, -);
solve(root);
}
}
}
int main(){
int n, cas = ;
while(~scanf("%d", &n)) {
memset(vis, , sizeof(vis));
for(int i=; i<=n; i++) scanf("%d", &col[i]);
for(int i=; i<=n; i++) mp[i].clear();
for(int i=; i<n; i++) {
int u,v;
scanf("%d%d", &u, &v);
mp[u].pb(v);
mp[v].pb(u);
} ans = ;
root = ; wt[] = inf;
curn = n;
findRoot(, -);
solve(root);
printf("Case #%d: %lld\n", ++cas, ans);
}
return ;
}
/*
6
1 2 3 1 2 3
1 2
1 3
3 4
3 5
4 6
*/

虚树 + 树上差分法:

  对于一种颜色,可以把树分割成许多联通块,同一个联通块内,这种颜色不会产生影响,所以某个点上,某个颜色的影响就是n - size,size是包含这个点的联通块的大小。

  由于有多种颜色,我们可以对每种颜色构建对应的虚树,选择这种颜色的点和这些点的直接儿子作为关键点。类似树上差分的思想,先把答案保存在每个联通块最上面的点。

  

#include <bits/stdc++.h>

using namespace std;
#define pb push_back
#define fi first
#define se second
#define debug(x) cerr<<#x << " := " << x << endl;
#define bug cerr<<"-----------------------"<<endl;
#define FOR(a, b, c) for(int a = b; a <= c; ++ a) typedef long long ll;
typedef long double ld;
typedef pair<int, int> pii;
typedef pair<ll, ll> pll;
typedef pair<pii, int>PII; template<class T> void _R(T &x) { cin >> x; }
void _R(int &x) { scanf("%d", &x); }
void _R(ll &x) { scanf("%lld", &x); }
void _R(double &x) { scanf("%lf", &x); }
void _R(char &x) { scanf(" %c", &x); }
void _R(char *x) { scanf("%s", x); }
void R() {}
template<class T, class... U> void R(T &head, U &... tail) { _R(head); R(tail...); } template<typename T>
inline T read(T&x){
x=;int f=;char ch=getchar();
while (ch<''||ch>'') f|=(ch=='-'),ch=getchar();
while (ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x=f?-x:x;
} const ll inf = 0x3f3f3f3f3f3f3f3f; const int mod = 1e9+; /**********showtime************/
const int maxn = 2e5+;
int col[maxn],vis[maxn];
vector<int>mp[maxn],xu_mp[maxn];
vector<int>node[maxn],xu;
int sz[maxn], dfn[maxn], dp[maxn], tim;
int fa[maxn][];
ll fen[maxn],ans; void dfs(int u, int o) {
sz[u] = ; dfn[u] = ++tim;
fa[u][] = o;
dp[u] = dp[o] + ;
for(int i=; i<; i++)
fa[u][i] = fa[fa[u][i-]][i-];
for(int v : mp[u]) {
if(v == o) continue;
dfs(v, u);
sz[u] += sz[v];
}
} int lca(int u, int v) {
if(dp[u] < dp[v]) swap(u, v); for(int i=; i>=; i--) {
if(dp[fa[u][i]] >= dp[v])
u = fa[u][i];
}
if(u == v) return u; for(int i=; i>=; i--) {
if(fa[u][i] != fa[v][i])
u = fa[u][i], v = fa[v][i];
}
return fa[u][];
} bool cmp(int x, int y) {
return dfn[x] < dfn[y];
}
int used[maxn];
int nsz[maxn];
int curcol;
int n;
int cdp[maxn];
//求虚树上每个联通块的大小
void gaoNewSz(int u, int o) {
ll s = ;
cdp[u] = ;
for(int v : xu_mp[u]) {
if(v == o) continue;
gaoNewSz(v, u);
if(col[v] == curcol)
cdp[u] += sz[v];
else cdp[u] += cdp[v];
}
nsz[u] = n - (sz[u] - cdp[u]);
}
//建立树上的差分
void gaoSub(int u, int fa, int val) {
int w = val;
if(col[u] == curcol) {
fen[u] -= val;
}
else if(col[fa] == curcol || u == )
{
fen[u] += nsz[u];
w = nsz[u];
} for(int v : xu_mp[u]) {
if(v == fa) continue;
if(col[u] == curcol)gaoSub(v, u, );
else gaoSub(v, u, w);
}
} //建立虚树
void build(vector <int> & xu) {
sort(xu.begin(), xu.end(), cmp);
stack<int>st;
queue<int>que; for(int i=; i<xu.size(); i++) {
int u = xu[i];
if(st.size() <= ) st.push(u);
else {
int x = st.top(); st.pop();
int o = lca(x, u);
if(o == x) {
st.push(x);
st.push(u);
continue;
}
while(!st.empty()) {
int y = st.top(); st.pop(); if(dfn[y] > dfn[o]) {
xu_mp[y].pb(x);
if(used[y] == ) used[y] = , que.push(y);
x = y;
}
else if(dfn[y] == dfn[o]) {
xu_mp[y].pb(x);
st.push(y);
if(used[y] == ) used[y] = , que.push(y);
break;
}
else {
xu_mp[o].pb(x);
st.push(y);
st.push(o);
if(used[o] == ) used[o] = , que.push(o);
break;
}
}
st.push(u);
}
}
while(st.size() > ) {
int u = st.top(); st.pop();
int v = st.top();
xu_mp[v].pb(u);
//xu_mp[u].pb(v);
// if(used[u] == 0) used[u] = 1, que.push(u);
if(used[v] == ) used[v] = , que.push(v);
}
while(!st.empty())st.pop(); gaoNewSz(, );
gaoSub(, , ); while(!que.empty()) {
int u = que.front();
xu_mp[u].clear();
used[u] = ;
que.pop();
}
} //树上差分,最后的更新
void pushdown(int u, int fa, ll val) {
ans += fen[u] + val + n;
val += fen[u];
for(int v : mp[u]) {
if(v == fa) continue;
pushdown(v, u, val);
}
} int main(){
int cas = ;
while(~scanf("%d", &n)){
ans = ;tim = ;
for(int i=; i<=n; i++){
mp[i].clear();
fen[i] = ;
vis[i] = ;
dp[i] = ;
node[i].clear();
}
for(int i=; i<=n; i++) {
read(col[i]);
vis[col[i]] = ;
node[col[i]].pb(i);
}
for(int i=; i<n; i++) {
int u,v;
read(u); read(v);
mp[u].pb(v);
mp[v].pb(u);
} dfs(, ); for(int i=; i<maxn; i++) {
if(vis[i]) {
xu.clear();
if(col[] != i) xu.pb();
for(int v : node[i]) {
xu.pb(v);
for(int k : mp[v]) {
if(col[k] != i && dp[k] > dp[v])
xu.pb(k);
}
}
curcol = i;
build(xu);
}
}
pushdown(, , );
printf("Case #%d: %lld\n", ++cas, (ans - n )/ );
}
return ;
}

附上虚树建立的网上流行模板

void insert(int x) {
if(top == ) {s[++top] = x; return ;}
int lca = LCA(x, s[top]);
if(lca == s[top]){ s[++top] = x;return ;}
while(top > && dfn[s[top - ]] >= dfn[lca]) add_edge(s[top - ], s[top]), top--;
if(lca != s[top]) add_edge(lca, s[top]), s[top] = lca;//
s[++top] = x;
}

【HDU6035】 Colorful Tree的更多相关文章

  1. 【hdu6035】 Colorful Tree dfs序

    题目传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6035 题目大意:给你一棵树,树上每个节点都有一个颜色. 现在定义两点间的距离为两点最短路径上颜色集合 ...

  2. 【BZOJ】2631: tree LCT

    [题意]给定n个点的树,每个点初始权值为1,m次操作:1.x到y的点加值,2.断一条边并连一条边,保证仍是树,3.x到y的点乘值,4.x到y的点权值和取模.n,m<=10^5. [算法]Link ...

  3. 【BZOJ2212】[Poi2011]Tree Rotations 线段树合并

    [BZOJ2212][Poi2011]Tree Rotations Description Byteasar the gardener is growing a rare tree called Ro ...

  4. 【题解】Digit Tree

    [题解]Digit Tree CodeForces - 716E 呵呵以为是数据结构题然后是淀粉质还行... 题目就是给你一颗有边权的树,问你有多少路径,把路径上的数字顺次写出来,是\(m\)的倍数. ...

  5. 【题解】[P4178 Tree]

    [题解]P4178 Tree 一道点分治模板好题 不知道是不是我见到的题目太少了,为什么这种题目都是暴力开值域的桶QAQ?? 问点对,考虑点分治吧.直接用值域树状数组开下来,统计的时候直接往树状数组里 ...

  6. 【总结】Link-Cut Tree

    这是一篇关于LCT的总结 加删边的好朋友--Link Cut Tree Link-Cut Tree,LCT的全称 可以说是从树剖引出的问题 树剖可以解决静态的修改或查询树的链上信息:那如果图会不断改变 ...

  7. 【BZOJ】1468: Tree(POJ1741) 点分治

    [题意]给定带边权树,求两点距离<=k的点对数.n<=40000. [算法]点分治 [题解]对于一个区域,选择其重心x作为根,则划分出来的每棵子树都是子区域,可以证明至多划分log n次( ...

  8. 【Manacher】Colorful String

    The value of a string s is equal to the number of different letters which appear in this string. You ...

  9. 【题解】【BT】【Leetcode】Binary Tree Preorder/Inorder/Postorder (Iterative Solution)

    [Inorder Traversal] Given a binary tree, return the inorder traversal of its nodes' values. For exam ...

随机推荐

  1. decode函数的几种用法

    1:使用decode判断字符串是否一样 DECODE(value,if1,then1,if2,then2,if3,then3,...,else) 含义为 IF 条件=值1 THEN RETURN(va ...

  2. 1. 源码分析---SOFARPC可扩展的机制SPI

    这几天离职在家,正好没事可以疯狂的输出一下,本来想写DUBBO的源码解析的,但是发现写DUBBO源码的太多了,所以找一个写的不那么多的框架,所以就选中SOFARPC这个框架了. SOFARPC是蚂蚁金 ...

  3. 【错误】【vscode】"'#' not expected here"

    今天使用vscode发现完整的代码报错了,但依然可以运行

  4. java web 加载Spring --web.xml 篇

    spring是目前最流行的框架.今天谈谈对spring的认识 起步 javaweb中我们首先会遇到的配置文件就是web.xml,这是javaweb为我们封装的逻辑,不在今天的研究中.略过,下面是一个标 ...

  5. Git使用(码云)

    1.安装git软件(码云/GitHub) 2.码云注册,保存代码 3.创建代码托管仓库,仓库名相当于码云上的文件夹 4.写作业并提交 在作业文件夹上,右键选择‘get bash here’ 在黑框里输 ...

  6. 【模板】珂朵莉树(ODT)(Codeforces 896C Willem, Chtholly and Seniorious)

    题意简述 维护一个数列,支持区间加,区间赋值,区间求第k小,区间求幂和 数据随机 题解思路 ODT是一种基于std::set的暴力数据结构. 每个节点对应一段区间,该区间内的数都相等. 核心操作spl ...

  7. cs231n---循环神经网络

    1 RNN介绍 (1)一对多,多对一,多对多的任务 传统的神经网络只能处理一对一的任务,而RNN可以处理一对多,多对一,多对多的任务: 其中,一些典型的应用如下: Image Captioning:i ...

  8. 《机器学习技法》---核型SVM

    (本文内容和图片来自林轩田老师<机器学习技法>) 1. 核技巧引入 如果要用SVM来做非线性的分类,我们采用的方法是将原来的特征空间映射到另一个更高维的空间,在这个更高维的空间做线性的SV ...

  9. Spring aop注解失效

    问题 在spring 中使用 @Transactional . @Cacheable 或 自定义 AOP 注解时,对象内部方法中调用该对象的其他使用aop机制的方法会失效. @Transactiona ...

  10. Windows Server 2008创建域环境

    介绍一下域环境搭建,域主要用于中大型企业,小型企业计算机数量不多,而中大型企业计算机比较多,使用域可以方便管理,安全性也比在工作组中安全 1.安装完操作系统默认都属于WORKGROUP工作组. 2.安 ...