[题解][YZOJ50113] 枇杷树

简要题意#

m 个操作,每次操作都会产生一个树的版本 (0 开始).

一次操作把 xi 版本的树的点 uyi 版本的树的点 v 连一条权值是 w 的边。( yi 上的全部点的编号加 sizxi

求每个版本:

i=1siz1j=i+1sizdis(i,j)

m300,u,v1018,siz2×1018

解题思路#

首先,将答案考虑为每条边乘上两边的子树大小。

然后,考虑产生的新版本 (记为 z ) 的答案 ansz 的组成:

  • 原来版本内的路径和:ansx+ansy
  • (u,v,w) 的贡献:sizx×sizy×w
  • 版本间的贡献 ( 记 fx(u) 为版本 x 中,所有点到 u 的路径和):fx(u)×sizy+fy(v)×sizx.

那么现在的主要问题就是每次如何得到 fx(u)。( 记 disx(u,v) 表示版本 x 中,uv 的路径长度)

还是分析当得到一个新的版本 z 时,fz(a) 的组成:

  • a 原来属于版本 xfx(a)+fy(v)+(w+disx(a,u))×sizy.
  • a 原来属于版本 yfy(a)+fx(u)+(w+disy(a,v))×sizx.

此时,fx(u)/fy(v) 算作已知,故只需考虑 disz(a,b) 如何计算:

  • a,b 都在版本 x/y,调用 disx/y(a,b);
  • a,b 不在同一版本,disx(a,u)+w+disy(bv).

此时整个问题在递归中解决了,现在来分析一下复杂度。

对于 fx(u),版本数是 O(m) 级别,u 的取值是 O(m) 级别,总的状态就是 O(m2) 级别,转移 O(1)

disx(a,b) 的计算平均是 O(1) 的,所以计算所有的 fx(u)O(m2) 的。

计算 ans 时, fx(u) 会被调用 O(m) 次,所以总的时间复杂度是 O(m3)

代码#

inline int Dis(int z, int a, int b){
	if(a == b) return 0;
	if(dis[z].find({a, b}) != dis[z].end()) return dis[z][{a, b}];
	int x = X[z], y = Y[z], w = W[z]; LL u = U[z], v = V[z];
	if(a <= siz[x] && b <= siz[x]) return dis[z][{a, b}] = Dis(x, a, b);
	if(siz[x] < a && siz[x] < b) return dis[z][{a, b}] = Dis(y, a - siz[x], b - siz[x]);
	int sum = w; if(a > siz[x]) swap(a, b);
	Plus(sum, Dis(x, a, u)), Plus(sum, Dis(y, b - siz[x], v));
	return dis[z][{a, b}] = sum;
}
inline int F(int z, LL a){
	if(z == 0 && a == 1) return 0;
	if(f[z].find(a) != f[z].end()) return f[z][a];
	int x = X[z], y = Y[z], w = W[z]; LL u = U[z], v = V[z];
	if(a <= siz[x]){
		int sum = 0;
		Plus(sum, F(x, a)), Plus(sum, F(y, v));
		Plus(sum, siz[y] % mod * Mod(w + Dis(x, a, u) - mod) % mod);
		return f[z][a] = sum;
	}
	int sum = 0;
	Plus(sum, F(y, a - siz[x])), Plus(sum, F(x, u));
	Plus(sum, siz[x] % mod * Mod(w + Dis(y, a - siz[x], v) - mod) % mod);
	return f[z][a] = sum;
}
inline void solve(int x, int y, LL u, LL v, int w, int z){
	X[z] = x, Y[z] = y, U[z] = u, V[z] = v, W[z] = w;
	Plus(Ans[z], Mod(Ans[x] + Ans[y] - mod));
	Plus(Ans[z], siz[x] % mod * (siz[y] % mod) % mod * w % mod);
	Plus(Ans[z], Mod(siz[y] % mod * F(x, u) % mod + siz[x] % mod * F(y, v) % mod - mod));
	siz[z] = siz[x] + siz[y]; 
}

参考#

https://www.cnblogs.com/qjbqjb/p/15905279.html

posted @   IrisT  阅读(51)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 使用C#创建一个MCP客户端
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示
主题色彩