首页 > 其他分享 >题解 BZOJ4543【[POI2014] HOT-Hotels】

题解 BZOJ4543【[POI2014] HOT-Hotels】

时间:2023-07-25 22:12:20浏览次数:57  
标签:长链 题解 len 儿子 height HOT BZOJ4543 son DP

长链剖分优化 DP 板子题了,但是虽然是板子这个转移方程也很难想。

problem

树。求 \(\sum_{1\leq i<j<k\leq n}[dist(i,j)=dist(i,k)=dist(j,k)].\)。\(n\leq 10^5\)。

solution

与重链剖分相似,长链剖分是将树剖成很多条长链。

我们定义长儿子,为一个点的儿子中子树深度最大的一个儿子。或者这样定义:

  • 给每个点一个 \(height_u\)。叶子节点的 \(height\) 为 \(1\)。
  • 则其它所有点的 \(height_u\) 定义为 \(\max_{v\in son}\{height_v\}+1\),并记录取到 \(height_u\) 的儿子 \(v\) 为长儿子(保留一个)
  • 长链就是一条到叶节点的链,满足链上任意一个父亲的儿子是它的长儿子。

有几个性质值得留意:

  • 所有长链长度总和为 \(O(n)\),因为每一个点都只在一个长链中。
  • 任意一个节点 \(x\) 的 \(k\) 级祖先 \(y\) 所在长链的长度一定大于等于 k,否则 \(y\) 所在长链应该接到 \(x\) 上去。
  • 从一个节点开始向上跳长链,最多跳 \(O(\sqrt{n})\) 次。因为每次跳到的长链长度一定是单调递增的。

长链剖分可以 \(O(n\log n)-O(1)\) 求 LCA、LA。

以下缩写 \(height_u\) 为 \(len_u\),如果 \(u\) 是长链顶,那么 \(len_u\) 是这条长链的点数。另外还有不等式 \(len_u\geq len_v+1\),证明请参见 \(len_u\) 的定义。


考虑这个题的 DP 怎么写,这里直接给结论了。

  • \(f(u,d)\) 表示点 \(u\) 的子树中离它距离为 \(d\) 的点。根据定义,\(0\leq d<len_u\),边界 \(f(u,0)=1\) 即它自己。
  • \(f(u,d)=\sum_v f(v,d-1)\)。
  • \(g(u,d)\) 表示,点 \(u\) 的子树中,有多少点对 \((i,j)\),使得 \(dist(i,k)=dist(j,k)=dist(k,u)+d\),其中 \(k=lca(i,j)\),感性的理解就是在 \(u\) 顶上接长度(边)为 \(d\) 的链后有多少个答案。同样的 \(0\leq d<len_u\)。
  • \(g(u,d)\) 的第一种转移来自它的所有儿子的 \(g(v,d+1)\) 之和。
  • 考虑 [一个儿子 \(v\) 的子树] 和 [\(v\) 加入前 \(u\) 维护的子树] 之间产生的答案,则 \(g(u,d):=g(u,d)+f(u,d)f(v,d-1)\)。
  • 考虑答案时,也是考虑 [一个儿子 \(v\) 的子树] 和 [\(v\) 加入前 \(u\) 维护的子树] 之间的贡献。当 [\(v\) 加入前 \(u\) 维护的子树] 为空时,答案加上 \(g(v,1)\)。反之,要么是 \(v\) 选两个,\(u\) 选一个,要么是 \(v\) 选一个,\(u\) 选两个,分别是 \(g(v,d+1)f(u,d)\) 和 \(f(v,d-1)g(u,d)\)。

到这里应该可以写一个 \(O(n^2)\) DP。考虑优化,因为这里的 \(f,g\) 都和深度有关,所以长链剖分优化,就是先 DP 长儿子,继承长儿子的信息,然后 DP 轻儿子,在合并轻儿子信息时的复杂度为 \(O(len_v)\),然后我们知道每条长链都只在头上被暴力,所有长链之和为 \(O(n)\),所以整个题就 \(O(n)\)。

看上去非常简单是吧!其实很难写,这里介绍指针写法,首先可能需要手写一下内存池,写法可供借鉴:

LL _[1<<19],*_pos=_+sizeof(_)/sizeof(LL);
LL* alloc(int sz){return _pos-=sz;}

然后考虑:在我 DP 长儿子的时候,我通过使得 f[son[u]]=f[u]+1,使得长儿子 DP 完后,原来的 f[son[u]][0] 变成 f[u][1](可能有点抽象,建议画出内存条),同理 g[son[u]]=g[u]-1。然后 DP 和刚才上面一样(注意转移顺序,\(ans\to g\to f\))。注意数组越界的问题,我们可以看一下正常写出来怎么样:

for(int i=0;i<hei[v];i++){
	ans+=h[u][i+1]*f[v][i];
	ans+=f[u][i-1]*h[v][i];
}
for(int i=0;i<hei[v];i++){
	h[u][i-1]+=h[v][i];
	h[u][i+1]+=f[u][i+1]*f[v][i];
	f[u][i+1]+=f[v][i];
}

这里注意如果我们枚举 \(0\leq i<len_v\) 的话,访问 \(f(u,i+1)\) 是没有问题的,因为 \(len_u\geq len_v+1\);但是 \(i-1\) 就有问题,我们要判断一下是否有 \(i\geq 1\)。

然后看一眼空间问题,\(f\) 只用开一倍空间是没问题的,但是我们要看一下 \(g\),结论是 \(g\) 要开两倍空间,向前 \(len_u\) 向后 \(len_u\),这是因为 g[son[u]]=g[u]-1,所以向前开 \(len_u\);又因为 \(g(u,d)\) 的 \(d\) 的定义域为 \(0\leq d<len_u\),顶上的 \(u\) 向后开 \(len_u\);所以就是这样开数组就够了。(其实这个东西每次传上来的时候会舍弃上一位的 \(0\),然后多算一个 \(len_u\))

那么这题就结束了。

Code

点击查看代码

Rename \(height,len\to hei\),\(g\to h\)。


#include <cstdio>
#include <vector>
#include <cstring>
#include <algorithm>
using namespace std;
#ifdef LOCAL
#define debug(...) fprintf(stderr,##__VA_ARGS__)
#else
#define debug(...) void(0)
#endif
typedef long long LL;
int n,hei[1<<17],son[1<<17];
vector<int> g[1<<17];
LL _[1<<20],*_pos=_+sizeof(_)/sizeof(LL);
LL* alloc(int sz){return _pos-=sz;}
LL*f[1<<17],*h[1<<17],ans=0;
void dfs(int u,int fa){
	for(int v:g[u]) if(v!=fa){
		dfs(v,u);
		if(hei[son[u]]<hei[v]) son[u]=v;
	}
	hei[u]=hei[son[u]]+1;
}
void dp(int u,int fa){
	if(son[u]) f[son[u]]=f[u]+1,h[son[u]]=h[u]-1,dp(son[u],u);
	f[u][0]++;
	ans+=h[u][0],debug("tr0,u=%d,ans+=%lld\n",u,h[u][0]);
	for(int v:g[u]) if(v!=fa&&v!=son[u]){
		f[v]=alloc(hei[v]),h[v]=alloc(hei[v]<<1)+hei[v],dp(v,u);
		for(int i=0;i<hei[v];i++){
			//hei[u]>=hei[v]+1
			ans+=h[u][i+1]*f[v][i],debug("tr1,u=%d,v=%d,i=%d,ans+=%lld*%lld\n",u,v,i,h[u][i+1],f[v][i]);
			if(i>=1) ans+=f[u][i-1]*h[v][i],debug("tr2,u=%d,v=%d,i=%d,ans+=%lld*%lld\n",u,v,i,f[u][i-1],h[v][i]);
		}
		for(int i=0;i<hei[v];i++){
			if(i>=1) h[u][i-1]+=h[v][i];
			h[u][i+1]+=f[u][i+1]*f[v][i];
			f[u][i+1]+=f[v][i];
		}
	}
	debug("son[%d]=%d,hei[%d]=%d\n",u,son[u],u,hei[u]);
	debug("f:"); for(int i=0;i<hei[u];i++) debug("%lld,",f[u][i]); debug("\n");
	debug("h:"); for(int i=0;i<hei[u];i++) debug("%lld,",h[u][i]); debug("\n");
}
int main(){
	scanf("%d",&n);
	for(int i=1,u,v;i<n;i++) scanf("%d%d",&u,&v),g[u].push_back(v),g[v].push_back(u);
	dfs(1,0);
	f[1]=alloc(hei[1]),h[1]=alloc(hei[1]<<1)+hei[1],dp(1,0);
	printf("%lld\n",ans);
	return 0;
}

标签:长链,题解,len,儿子,height,HOT,BZOJ4543,son,DP
From: https://www.cnblogs.com/caijianhong/p/solution-BZOJ4543.html

相关文章

  • Codeforces 1852A Ntarsis' Set 题解
    题目传送门:Codeforces1852ANtarsis'Set题意给定一个集合,里面初始有\(1,2,3...10^{1000}\),告诉你每天会拿掉其中的第\(a_1,a_2,a_3...a_n\)个,询问这样的\(k\)天之后剩下的最小的数是多少。分析思考如果\(x\)在这天没有被删掉,那么哪些被删掉的位置会对它产生什么......
  • CF1776M Parmigiana With Seafood 题解
    先将所有的叶子取\(\max\)贡献给答案,以下讨论的所有点中不考虑叶子。首先可以考虑先手能否删到\(n\):不难发现当\(2\midn\)的时候可以,然后我们就排除了一半的\(n\),于是以下令\(2\not\midn\)。接下来,考虑先手能否删掉\(n-1\),那么把\(n-1\ton\)的路径当成一个大点,......
  • 洛谷 P9221 「TAOI-1」Pentiment 题解
    Description给定\(n\timesm\)的矩阵,从第\(1\)行任意格子出发,每次向下、左、有走一步,有\(q\)个障碍不能经过,求走到第\(n\)行任意格子的方案。对于所有数据,\(1\leqn,m\leq10^9\),\(1\leqq\leq10^5\)。link:https://www.luogu.com.cn/problem/P9221Solution算法一考......
  • CodeForces 725F Family Photos
    洛谷传送门CF传送门不错的贪心题。我们考虑一对照片只有一张的情况。那么先后手会按照\(a+b\)降序取。因为若\(a_i+b_i\gea_j+b_j\),变形得\(a_i-b_j\gea_j-b_i\)且\(b_i-a_j\geb_j-a_i\),所以对于双方先取\(i\)一定是不劣的。回到原题,如果\(a_{i......
  • 题解 LGP2300【合并神犇】
    Problem随机\(n\)个正整数组成序列。将序列分尽量多的段数,使得前一段的和不大于后一段的和。求能分成多少段。输出\(n-ans\)。\(n\leq10^5\),值域不重要。Solution状态设计为:\(f_i=1+\min_{sum_i-sum_j\geqg_j}f_j\)表示前\(i\)个数字划分的最多段数,\(g_j\)定义为\(f_......
  • 洛谷P3629 [APIO2010] 巡逻题解
    题目链接P3629[APIO2010]巡逻-洛谷|计算机科学教育新生态(luogu.com.cn)思路n个村庄,n-1条道路,原图为树1.若k=0(不修建道路)那么答案为(n-1)*2 每个道路会走两遍2.若k为1(修建一条道路)设修建的道路(r1)所在的环长度为L那么答案为(n-1)*2-L+2可以看到r1所在环的道路只走了......
  • 题解 P1150 【Peter的烟】
    postedon2020-11-1410:00:20|under题解|source2023编者注:本篇题解的方法过于暴力,但是尊重历史。请不要太在意。—-教你们用栈做这道题原题传送门看到这题,第一反应是用stack做。我们可以把Peter手上的烟看作一个栈,一根烟就是一个元素,抽了\(n\)支烟就从栈里pop几个,......
  • 题解 P1008 【三连击】
    postedon2020-11-1217:25:10|under题解|source2023编者注:请尊重历史。本题正解是暴力枚举先引用我们老师的一句话:(无恶意)不会吧不会吧,不会还有人不会写三连击吧!废话不多说,开始解题:理解题目和做题思路P1008三连击题目链接:https://www.luogu.com.cn/problem/......
  • 题解 CF1501A 【Alexey and Train】
    postedon2021-03-1321:57:02|under题解|source简单模拟题,考验选手的读题能力和使用谷歌翻译的能力。先定义一个\(now=0\),我们最后算出来的结果为\(now\)。对于每个站(不包括第\(n\)站),我们需要加上\(3\)个部分:额外时间,now+=ex[i];第\(i-1\)站到第\(i\)站的时......
  • 题解 CF1501B 【Napoleon Cake】
    postedon2021-03-1617:42:06|under题解|source题目可以转化一下:给一个长为\(n\)的数组\(a\),请求出一个长为\(n\)的数组\(b\)。要求若\(a_i\)不为\(0\),则\([b_{i-a_i+1},b_i]\)这个区间要为\(1\)。知道这个题目意思就好办了。题目说\(n\leq2\times10^5\),......