应该是一道中等难度的点分?麻烦在一些细节。

题目描述

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

现在他想让你求出所有的sum[i]

输入输出格式

输入格式:

第一行为一个整数n,表示树节点的数量

第二行为n个整数,分别表示n个节点的颜色c[1],c[2]……c[n]

接下来n-1行,每行为两个整数x,y,表示x和y之间有一条边

输出格式:

输出n行,第i行为sum[i]

说明

sum[1]=s(1,1)+s(1,2)+s(1,3)+s(1,4)+s(1,5)=1+2+3+2+2=10
sum[2]=s(2,1)+s(2,2)+s(2,3)+s(2,4)+s(2,5)=2+1+2+1+3=9
sum[3]=s(3,1)+s(3,2)+s(3,3)+s(3,4)+s(3,5)=3+2+1+2+3=11
sum[4]=s(4,1)+s(4,2)+s(4,3)+s(4,4)+s(4,5)=2+1+2+1+3=9
sum[5]=s(5,1)+s(5,2)+s(5,3)+s(5,4)+s(5,5)=2+3+3+3+1=12

对于40%的数据,n<=2000

对于100%的数据,1<=n,c[i]<=10^5


题目分析

想法一:按颜色拆贡献

这里应该是有一种小颜色大颜色的分块套路的。但是这个想法我只能解决全局路径的数量和,并不会落实到点的询问。

想法二:点分治

目前尚未归结出点分治适用的具体问题范围……不过这一题是可以用点分解决的。

考虑每一层点分树,我们只需要对它的节点处理贡献。这里的贡献分为两部分:重心答案;经过重心的路径对子树的贡献。

重心的答案只需要以它自身为根,遍历一边该层点分树即可。子树内的答案处理要略微麻烦一些,需要分颜色来考虑贡献。记$colCnt[i]$为所有以重心为起点的路径中,含有颜色$i$的路径条数。然后首先假定子树内所有点的答案都为$\sum colCnt[i]$,再容斥考虑重心到子树路径上的颜色所产生的贡献。

记当前点分树中除去正在处理的子树的大小为$outTot$,那么对于子树内点$x$,由于它具有颜色$c[x]$,所以对自身的答案有一个$outTot-colCnt[c[x]]$的贡献。并且,这一个贡献对于$x$的子树也是一概适用的,所以这一个标记要差分式地下传。

整体思路就是这些。这一题的点分涉及到例如“子树结构的重定向”或是“两个颜色桶并存”的一些细节问题,所以实现上面可能有一定的难度。

(话说这题的码风怎么这么丑)

 #include<bits/stdc++.h>
typedef long long ll;
const int maxn = ;
const int maxm = ; ll ans[maxn];
int n,bloTot,outTot,c[maxn];
int size[maxn],son[maxn],root;
int edgeTot,head[maxn],nxt[maxm],edges[maxm];
int cols,cnt,cl,col[maxn],colTmp[maxn],colTim[maxn],colCnt[maxn],subCnt[maxn];
bool colEx[maxn],divEx[maxn]; int read()
{
char ch = getchar();
int num = , fl = ;
for (; !isdigit(ch); ch=getchar())
if (ch=='-') fl = -;
for (; isdigit(ch); ch=getchar())
num = (num<<)+(num<<)+ch-;
return num*fl;
}
void addedge(int u, int v)
{
edges[++edgeTot] = v, nxt[edgeTot] = head[u], head[u] = edgeTot;
edges[++edgeTot] = u, nxt[edgeTot] = head[v], head[v] = edgeTot;
}
void getRoot(int x, int fa)
{
size[x] = , son[x] = ;
for (int i=head[x]; i!=-; i=nxt[i])
{
int v = edges[i];
if (divEx[v]||v==fa) continue;
getRoot(v, x), size[x] += size[v];
son[x] = std::max(son[x], size[v]);
}
son[x] = std::max(son[x], bloTot-size[x]);
if (son[x] < son[root]) root = x;
}
void colDfs(int x, int fa, int *cnt)
{
if (!colEx[c[x]]) colEx[c[x]] = , col[++cols] = c[x];
if ((++colTim[c[x]])==) cnt[c[x]] += size[x];
for (int i=head[x]; i!=-; i=nxt[i])
if ((!divEx[edges[i]])&&(edges[i]!=fa))
colDfs(edges[i], x, cnt);
--colTim[c[x]];
}
void colClear()
{
for (int i=; i<=cl; i++) colEx[colTmp[i]] = ;
cols = ;
}
void modify(int x, int fa, ll tag)
{
if ((++colTim[c[x]])==) tag += outTot-colCnt[c[x]];
ans[x] += tag+cnt;
for (int i=head[x]; i!=-; i=nxt[i])
{
int v = edges[i];
if (v==fa||divEx[v]) continue;
modify(v, x, tag);
}
--colTim[c[x]];
}
void calc(int rt)      //核心操作在这里
{
colClear(), getRoot(rt, );
colDfs(rt, , colCnt);
cnt = , cl = cols;
for (int i=; i<=cols; i++)
cnt += colCnt[col[i]], colTmp[i] = col[i];
ans[rt] += cnt;
for (int i=head[rt]; i!=-; i=nxt[i])
{
int v = edges[i];
if (divEx[v]) continue;
for (int j=; j<=cl; j++) subCnt[colTmp[j]] = ;  //及时清除数组
colClear();
colEx[c[rt]] = ;
colDfs(v, rt, subCnt);          //统计子树内的含颜色i路径条数
colEx[c[rt]] = ;
colCnt[c[rt]] -= size[v], cnt -= size[v];   //除去重心出发的路径
for (int j=; j<=cols; j++)
{
colCnt[col[j]] -= subCnt[col[j]];  //除去子树内的路径(因为考虑子树外路径)
cnt -= subCnt[col[j]];
}
outTot = size[rt]-size[v], modify(v, rt, );  //对子树内累加贡献
colCnt[c[rt]] += size[v], cnt += size[v];    //恢复处理子树前状态
for (int j=; j<=cols; j++)
{
colCnt[col[j]] += subCnt[col[j]];
cnt += subCnt[col[j]];
}
}
for (int i=; i<=cl; i++)
colCnt[colTmp[i]] = ;      //colTmp[]的作用;清空colCnt[]
}
void deal(int rt)
{
calc(rt), divEx[rt] = ;
for (int i=head[rt]; i!=-; i=nxt[i])
{
int v = edges[i];
if (divEx[v]) continue;
root = , bloTot = size[v];
getRoot(v, ), deal(root);
}
}
int main()
{
memset(head, -, sizeof head);
n = read(), son[] = n;
for (int i=; i<=n; i++) c[i] = read();
for (int i=; i<n; i++) addedge(read(), read());
bloTot = n, getRoot(, ), deal(root);
for (int i=; i<=n; i++) printf("%lld\n",ans[i]);
return ;
}

END

【点分治】luoguP2664 树上游戏的更多相关文章

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

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

  2. luoguP2664 树上游戏

    https://www.luogu.org/problemnew/show/P2664 考虑对于每种颜色包含的点和这些点的子节点建出虚树,发现只要将一个联通块中的东西 Dp + 差分一下就行了 当然要 ...

  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. bzoj4182/luoguP6326 Shopping(点分治,树上背包)

    bzoj4182/luoguP6326 Shopping(点分治,树上背包) bzoj它爆炸了. luogu 题解时间 如果直接暴力背包,转移复杂度是 $ m^{2} $ . 考虑改成点分治. 那么问 ...

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

    题意 题目链接 Sol 神仙题..Orz yyb 考虑点分治,那么每次我们只需要统计以当前点为\(LCA\)的点对之间的贡献以及\(LCA\)到所有点的贡献. 一个很神仙的思路是,对于任意两个点对的路 ...

随机推荐

  1. 1.函数的结构,调用,传参,形参,实参,args,kwargs,名称空间,高阶函数

    1.函数的初识 初始函数 获取任意一个字符串的元素的个数 s1='dsjdkjkfefenga' count=0 for i in s1: count+=1 print(count) 获取列表的元素的 ...

  2. 跟踪记录ABAP对外部系统的RFC通信

    对SAP系统而言,RFC最常见的系统间通信方式,SAP与SAP系统及SAP与非SAP系统之间的连接都可以使用它.它的使用便利,功能强大,在各种接口技术中,往往是最受(ABAP开发者)青睐的选择. 查询 ...

  3. Codecraft-17 and Codeforces Round #391 (Div. 1 + Div. 2, combined) C

    It's that time of the year, Felicity is around the corner and you can see people celebrating all aro ...

  4. 【手撸一个ORM】第四步、Expression(表达式目录树)扩展

    到这里,Orm的基架已经搭起来了,接下来就是激动人心的部分,表达式目录树转Sql语句,SqlDataReader转数据实体等等,但是在这之前,我们需要扩展下表达式目录树的方法,以方便后面的相关操作. ...

  5. 关于原生javascript的this,this真是个强大的东东

    最近一直坐在东钿微信服务平台,上上级领导提出一个要求,就是微信分享. 因为首页是一个tab切换页,领导想在分享的时候区分上产调还是评估.我研究了很久很久,一直都是失败,今天领导又问了.于是我就向我们老 ...

  6. 新项目升级到JFinal3.5之后的改变-着重体验自动依赖注入

    最近,JFinal3.5发布,喜大普奔,我也应JBolt用户的需求,将JBolt进行了升级,实现可配置自动注入开启,支持JFinal3.5的项目生成.具体可以看:JBolt升级日志 这等工作做完后,我 ...

  7. uvm_reg_fifo——寄存器模型(十五)

    当我们对寄存器register, 存储器memory, 都进行了建模,是时候对FIFO进行建模了 uvm_reg_fifo毫无旁贷底承担起了这个责任,包括:set, get, update, read ...

  8. win7双网卡走哪个网卡路由设置

    有没有软件能做这个我还真不知道.说说我的做法吧: 单位里无线是可以访问Internet的,有线是用来访问公司内部系统的. 默认的54M无线网络和100M的有线网络,系统在选择默认路由的时候肯定是选择有 ...

  9. SQL Server 填充因子

    在创建聚集索引时,表中的数据按照索引列中的值的顺序存储在数据库的数据页中.在表中插入新的数据行或更改索引列中的值时,Microsoft®   SQL   Server™   2000   可能必须重新 ...

  10. jacvaSe-LinkedList

    package com.java.chap08.sec02; import java.util.LinkedList; public class TestLinkedList { private st ...