https://www.luogu.org/problemnew/show/P2664

考虑对于每种颜色包含的点和这些点的子节点建出虚树,发现只要将一个联通块中的东西 Dp + 差分一下就行了

当然要考虑哪些东西要被加进去

如果把不是一个颜色的联通块放在一起加,里面就要算上 n - 联通块大小的贡献(画个图就行了

然后输出的时候每个点的贡献要 + n (因为自己对任何一个点的连边肯定包含自己这种颜色

博主差分的时候写挂了导致要 #define int long long,而且常数巨大

#include <bits/stdc++.h>
#define int long long
using namespace std; const int N = 1e5 + 5, LG = 17; vector <int> col[N], G[N], mdf[N], G2[N];
int pre[N][LG + 1], dep[N], sta[N], a[N], s[N], id[N], siz[N], book[N], f[N], h[N * 2];
int n, len, dfn, maxn, k; void init(int u, int fa) {
pre[u][0] = fa; dep[u] = dep[fa] + 1; id[u] = ++dfn; siz[u] = 1;
for(int i = 1; i <= LG; i++) pre[u][i] = pre[pre[u][i - 1]][i - 1];
for(vector <int> :: iterator it = G[u].begin(); it != G[u].end(); it++)
if(*it != fa) init(*it, u), siz[u] += siz[*it];
} int jump(int x, int k) {
for(int i = LG; i >= 0; i--)
if(k & (1 << i))
x = pre[x][i];
return x;
} int LCA(int x, int y) {
if(dep[x] > dep[y]) swap(x, y);
y = jump(y, dep[y] - dep[x]);
if(x == y) return x;
for(int i = LG; i >= 0; i--)
if(pre[x][i] != pre[y][i])
x = pre[x][i], y = pre[y][i];
return pre[x][0];
} bool cmp(int x, int y) {return id[x] < id[y];} void dfs1(int u, int top) {
if(book[u] == 0 && G[u].size() == 0) {
f[u] = siz[u]; return;
}
if(book[u] == 1) {
f[u] = 0;
for(vector <int> :: iterator it = G[u].begin(); it != G[u].end(); it++) {
int len = (dep[*it] - dep[u] - 1);
int son = jump(*it, len);
if(book[*it] == 1) {
dfs1(*it, top);
int sz = siz[son] - siz[*it];
s[son] += (n - sz); s[*it] -= (n - sz);
} else {
mdf[son].clear(); dfs1(*it, son);
int sz = siz[son] - siz[*it];
int allsz = sz + f[*it];
s[son] += (n - allsz);
for(vector <int> :: iterator itt = mdf[son].begin(); itt != mdf[son].end(); itt++) s[*itt] -= (n - allsz);
}
}
} else {
f[u] = siz[u];
for(vector <int> :: iterator it = G[u].begin(); it != G[u].end(); it++) {
int len = (dep[*it] - dep[u] - 1);
int son = jump(*it, len);
dfs1(*it, top);
if(book[*it] == 1) {
mdf[top].push_back(*it);
f[u] -= siz[*it];
} else {
f[u] -= (siz[*it] - f[*it]);
}
}
}
} void dfs2(int u, int fa) {
s[u] += s[fa];
for(vector <int> :: iterator it = G2[u].begin(); it != G2[u].end(); it++) {
if(*it != fa) dfs2(*it, u);
}
} signed main() {
cin >> n;
for(int i = 1; i <= n; i++) {
scanf("%lld", &a[i]);
col[a[i]].push_back(i);
maxn = max(maxn, a[i]);
}
for(int i = 1; i < n; i++) {
int a, b;
scanf("%lld %lld", &a, &b);
G[a].push_back(b);
G[b].push_back(a);
G2[a].push_back(b);
G2[b].push_back(a);
}
init(1, 0);
G[n + 1].push_back(1); book[n + 1] = 1;
for(int i = 1; i <= maxn; i++) {
k = col[i].size();
for(int j = 0; j < k; j++) h[j + 1] = col[i][j], book[col[i][j]] = 1;
int tmp = k; bool have = 0;
for(int j = 1; j <= tmp; j++) {
int u = h[j];
if(u == 1) have = 1;
for(vector <int> :: iterator it = G2[u].begin(); it != G2[u].end(); it++) {
if(*it != pre[u][0]) h[++k] = *it;
}
}
if(!have) {
for(vector <int> :: iterator it = G2[1].begin(); it != G2[1].end(); it++) {
h[++k] = *it;
}
}
sort(h + 1, h + k + 1, cmp);
k = unique(h + 1, h + k + 1) - h - 1;
sort(h + 1, h + k + 1, cmp);
sta[len = 1] = 1; G[1].clear();
for(int j = 1; j <= k; j++) {
if(h[j] == 1) continue;
int lca = LCA(h[j], sta[len]);
if(lca != sta[len]) {
while(id[lca] < id[sta[len - 1]]) {
G[sta[len - 1]].push_back(sta[len]);
len--;
}
if(id[lca] > id[sta[len - 1]]) {
G[lca].clear();
G[lca].push_back(sta[len]);
sta[len] = lca;
} else G[lca].push_back(sta[len]), len--;
}
G[h[j]].clear(); sta[++len] = h[j];
}
for(int j = 1; j < len; j++) G[sta[j]].push_back(sta[j + 1]);
dfs1(n + 1, 0); for(int j = 0; j < tmp; j++) book[col[i][j]] = 0;
}
dfs2(1, 0);
for(int i = 1; i <= n; i++) printf("%lld\n", s[i] + n);
return 0;
}

luoguP2664 树上游戏的更多相关文章

  1. luoguP2664树上游戏(点分治)

    题目链接:https://www.luogu.org/problem/P2664 题意:给定一颗带点权的树,结点数n<=1e5,点权<=1e5,用s(i,j)表示从i到j的路径上不同点权数 ...

  2. 【点分治】luoguP2664 树上游戏

    应该是一道中等难度的点分?麻烦在一些细节. 题目描述 lrb有一棵树,树的每个节点有个颜色.给一个长度为n的颜色序列,定义s(i,j) 为i 到j 的颜色数量.以及 现在他想让你求出所有的sum[i] ...

  3. 【Luogu2664】树上游戏(点分治)

    [Luogu2664]树上游戏(点分治) 题面 洛谷 题解 很好的一道点分治题. 首先直接点分治,考虑过每个分治重心的链的贡献. 我们从分治重心开始找每种颜色,强制令一种颜色只在其到分治重心的链上第一 ...

  4. 洛谷 P2664 树上游戏 解题报告

    P2664 树上游戏 题目描述 \(\text{lrb}\)有一棵树,树的每个节点有个颜色.给一个长度为\(n\)的颜色序列,定义\(s(i,j)\) 为 \(i\) 到 \(j\) 的颜色数量.以及 ...

  5. P2664 树上游戏

    P2664 树上游戏 https://www.luogu.org/problemnew/show/P2664 分析: 点分治. 首先关于答案的统计转化成计算每个颜色的贡献. 1.计算从根出发的路径的答 ...

  6. Luogu P2664 树上游戏 dfs+树上统计

    题目: P2664 树上游戏 分析: 本来是练习点分治的时候看到了这道题.无意中发现题解中有一种方法可以O(N)解决这道题,就去膜拜了一下. 这个方法是,假如对于某一种颜色,将所有这种颜色的点全部删去 ...

  7. LG2664 树上游戏

    树上游戏 题目描述 lrb有一棵树,树的每个节点有个颜色.给一个长度为n的颜色序列,定义s(i,j) 为i 到j 的颜色数量.以及 $$sum_i=\sum_{j=1}^ns(i,j)$$ 现在他想让 ...

  8. 【Luogu P2664】树上游戏

    Problem Description \(lrb\) 有一棵树,树的每个节点有个颜色.给一个长度为 \(n\) 的颜色序列,定义 \(s(i,j)\) 为 \(i\) 到 \(j\) 的颜色数量.以 ...

  9. 洛谷P2664 树上游戏(点分治)

    传送门 题解 因为一个sb错误调了一个晚上……鬼晓得我为什么$solve(rt)$会写成$solve(v)$啊!!!一个$O(logn)$被我硬生生写成$O(n)$了竟然还能过$5$个点……话说还一直 ...

随机推荐

  1. 安装glibc-2.14

    下载glibc-2.14.tar.gz 解压: [root@jrgc130 software]# tar xf glibc-2.14.tar.gz [root@jrgc130 software]# c ...

  2. 什么是jsonp?——使用jsonp解决跨域请求问题

    我们在使用ajax请求的时候经常会产生跨域问题,这是由于浏览器的同源策略导致的.所谓同源,即域名.协议.端口均相同,否则不管是静态页面还是动态网页或者web服务都无法通过ajax正常请求.有时候,我们 ...

  3. In function 'int av_clipl_int32_c(int64_t)': error: 'UINT64_C' was not declared in this scope

    cygwin下使用ndk编译jni时遇到的错误: /ffmpeg/include/libavutil/common.h: In function 'int av_clipl_int32_c(int64 ...

  4. 【HDU6026】Deleting Edges

    题意 有一个n个节点的无向图,结点编号从0-n-1,每条边的长度时1to9的一个正整数.现在要删除一些边(或者不删),使得到的新图满足下面两个要求. 1.新图是一颗树有n-1条边2.对于每个结点v(0 ...

  5. 136. Single Number唯一的一个只出现了一次的数字

    [抄题]: Given a non-empty array of integers, every element appears twice except for one. Find that sin ...

  6. c语言实践 创建两个包含8个元素的double类型数组,第二个元素的每个元素的值都是对应前一个元素的前n个元素的和

    意思就是第二个元素的num[2]等于第一个元素的num[0]+num[1]+num[2] #define COUNT 8 int main(void) { double num1[COUNT]; do ...

  7. c语言实践:求两个数的最大公约数

    我的思路是这样的:比如12和16这两个数.先理解一下概念,什么叫最大公约数.就是12有很多个因数,16也有很多个因数,这两堆因数中有一些重合的因数,在这些重合的因数中找到那个最大的.那么最大公约数一定 ...

  8. 3、python的传入参数

    转载:https://blog.csdn.net/abc_12366/article/details/79627263 1.位置参数: def func(a, b): print(a+b) func( ...

  9. 黑盒测试实践-任务进度-Day02

    使用工具 selenium 小组成员 华同学.郭同学.穆同学.沈同学.覃同学.刘同学 任务进度 在经过了昨天的基本任务分配之后,今天大家就开始了各自的内容,以下是大家任务的进度情况汇总. 华同学(任务 ...

  10. Monkey测试异常信息解读

    查看包名 1.cmd 下面输入 adb locat > D:\test.txt 2.ctrl+c 停掉刚刚 1 运行的进程 3.打开test.txt文件--搜索  Displayed  对应的内 ...