[题解][YZOJ50113] 枇杷树
简要题意
\(m\) 个操作,每次操作都会产生一个树的版本 \((\)从 \(0\) 开始\()\).
一次操作把 \(x_i\) 版本的树的点 \(u\) 和 \(y_i\) 版本的树的点 \(v\) 连一条权值是 \(w\) 的边。\((\) \(y_i\) 上的全部点的编号加 \(siz_{x_i}\) \()\)
求每个版本:
\]
\(m\le 300,u,v\leq 10^{18}, siz \leq 2 \times 10 ^{18}\)
解题思路
首先,将答案考虑为每条边乘上两边的子树大小。
然后,考虑产生的新版本 (记为 \(z\) ) 的答案 \(ans_z\) 的组成:
- 原来版本内的路径和:\(ans_x+ans_y\);
- 边 \((u,v,w)\) 的贡献:\(siz_x\times siz_y\times w\);
- 版本间的贡献 ( 记 \(f_x(u)\) 为版本 \(x\) 中,所有点到 \(u\) 的路径和):\(f_x(u)\times siz_y+f_y(v)\times siz_x\).
那么现在的主要问题就是每次如何得到 \(f_x(u)\)。( 记 \(dis_x(u,v)\) 表示版本 \(x\) 中,\(u\) 到 \(v\) 的路径长度)
还是分析当得到一个新的版本 \(z\) 时,\(f_z(a)\) 的组成:
- 若 \(a\) 原来属于版本 \(x\),\(f_x(a)+f_y(v)+(w+dis_x(a, u))\times siz_y\).
- 若 \(a\) 原来属于版本 \(y\),\(f_y(a)+f_x(u)+(w+dis_y(a, v))\times siz_x\).
此时,\(f_x(u)/f_y(v)\) 算作已知,故只需考虑 \(dis_z(a,b)\) 如何计算:
- \(a,b\) 都在版本 \(x/y\),调用 \(dis_{x/y}(a,b)\);
- \(a,b\) 不在同一版本,\(dis_x(a,u)+w+dis_y(b_v)\).
此时整个问题在递归中解决了,现在来分析一下复杂度。
对于 \(f_x(u)\),版本数是 \(O(m)\) 级别,\(u\) 的取值是 \(O(m)\) 级别,总的状态就是 \(O(m^2)\) 级别,转移 \(O(1)\)。
\(dis_x(a, b)\) 的计算平均是 \(O(1)\) 的,所以计算所有的 \(f_x(u)\) 是 \(O(m^2)\) 的。
计算 \(ans\) 时, \(f_x(u)\) 会被调用 \(O(m)\) 次,所以总的时间复杂度是 \(O(m^3)\)。
代码
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
[题解][YZOJ50113] 枇杷树的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
- 网络流n题 题解
学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...
- CF100965C题解..
求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...
随机推荐
- Hashmap如何同步?
当我们需要一个同步的HashMap时,有两种选择: ●使用Collections.synchronizedMap(..)来同步HashMap. ●使用ConcurrentHashMap的 这两个选项之 ...
- 学习FastDfs(二)
分布式文件系统(DFS) 指文件系统管理的物理存储资源不一定直接连接在本地节点上 而是通过计算机网络与节点相连 分布式文件系统的设计基于客户机/服务器模式 一个典型的网络可能包括多个多个用户访问的服务 ...
- gateway聚合swagger3统一管理api文档
springboot微服务整合swagger3方法很简单,下文会演示.但是在分布式项目中如果每个微服务都需要单独的分开访问获取接口文档就不方便了,本文将详细讲解springcloud gateway网 ...
- Linux基础学习 | 目录及文件
一.目录结构 Linux目录采用树形结构,以根目录/向下延伸呈一个倒置的树的形状. 每个目录下都有其相对应的子目录,而子目录中又有其子目录的存在,就像一棵树的树枝分叉,所有的目录层次结构分明,每个目录 ...
- simulink仿真过程
Simulink求解器 Simulink仿真过程 Simulink 模型的执行分几个阶段进行.首先进行的是初始化阶段,在此阶段,Simulink 将库块合并到模型中来,确定传送宽度.数据类型和采样时间 ...
- CSS3 用border写 空心三角箭头 (两种写法)
之前一直在寻找这种空心三角箭头, 终于知道了原理! 自己记录一下,顺便分享给之前跟我一样想要的撸友们~ 第一种写法 利用常见的 after伪元素 <!DOCTYPE html> <h ...
- canvas离屏、旋转效果实践——旋转的雪花
效果展示理论基础--"常见的canvas优化--模糊问题.旋转效果" 用离屏canvas画基础部分 1.封装画线函数 function drawLine(ctx,x1,y1,x2, ...
- h5 ios输入框与键盘 兼容性优化
起因 h5的输入框引起键盘导致体验不好,目前就算微信.知乎.百度等产品也没有很好的技术方案实现,尤其底部固定位置的输入框各种方案都用的前提下体验也并没有很好,这个问题也是老大难问题了.目前在准备一套与 ...
- Sentry前端部署拓展篇(sourcemap关联、issue关联、release控制)
原文首发于我的个人博客: https://lonhon.top/ 之前的<基础篇>主要介绍了Sentry和基本部署流程,在实际使用过程中你会发现Sentry受欢迎的原因:除了单纯的监控异常 ...
- es5语法下,javascript如何判断函数是new还是()调用
es5语法没有支持类class,但是可以通关函数来申明一个类,如下: function Person(name){ this.name=name; } var john=new Person('joh ...