简要题意

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

一次操作把 \(x_i\) 版本的树的点 \(u\) 和 \(y_i\) 版本的树的点 \(v\) 连一条权值是 \(w\) 的边。\((\) \(y_i\) 上的全部点的编号加 \(siz_{x_i}\) \()\)

求每个版本:

\[\sum_{i = 1}^{siz - 1} \sum_{j = i + 1}^{siz} dis(i, j)
\]

\(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] 枇杷树的更多相关文章

  1. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  2. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  3. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  4. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  5. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  6. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  7. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  8. 网络流n题 题解

    学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...

  9. CF100965C题解..

    求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...

随机推荐

  1. Kafka 缺点?

    由于是批量发送,数据并非真正的实时: 对于mqtt协议不支持: 不支持物联网传感数据直接接入: 仅支持统一分区内消息有序,无法实现全局消息有序: 监控不完善,需要安装插件: 依赖zookeeper进行 ...

  2. Mybatis是如何将sql执行结果封装为目标对象并返回的? 都有哪些映射形式?

    第一种是使用<resultMap>标签,逐一定义数据库列名和对象属性名之间的映 射关系. 第二种是使用 sql 列的别名功能,将列的别名书写为对象属性名. 有了列名与属性名的映射关系后,M ...

  3. NOW()和 CURRENT_DATE()有什么区别?

    NOW()命令用于显示当前年份,月份,日期,小时,分钟和秒. CURRENT_DATE()仅显示当前年份,月份和日期.

  4. Eclipse创建Spring XML配置文件插件

    引用:https://www.cnblogs.com/lideqiang/p/9067219.html 第一步:在 Eclipse Marketplace仓库中,搜索sts 第二步:安装Spring ...

  5. elasticsearch 的倒排索引是什么 ?

    解答:通俗解释一下就可以. 传统的我们的检索是通过文章,逐个遍历找到对应关键词的位置. 而倒排索引,是通过分词策略,形成了词和文章的映射关系表,这种词典+映射表 即为倒排索引. 有了倒排索引,就能实现 ...

  6. GC和GC Tuning

    GC和GC Tuning GC的基础知识 什么是垃圾 C语言申请内存:malloc free C++: new delete c/C++ 手动回收内存 Java: new ? 自动内存回收,编程上简单 ...

  7. 纯CSS实现柱形图

    CSS在处理排版之强大,没有做不到,只有想不到.下面我们将一同实现一个柱状图. 先打好一个具体的框架.我们利用无序列表做整体,里面的东西我们根本选择内联无素span,strong,em来填充. < ...

  8. 富文本编辑器CKeditor的配置和图片上传,看完不后悔

    CKeditor是一款富文本编辑器,本文将用极为简单的方式介绍一下它的使用和困扰大家很久的图片上传问题,要有耐心. 第一步:如何使用 1.官网下载https://ckeditor.com/ckedit ...

  9. 软件构造实验-JFinal

    导入JFinal的demo 可以增删改查 根据demo以及自己的理解,使用JFinal实现学生信息管理系统.

  10. MySQL外键约束On Delete和On Update的使用

    On Delete和On Update都有Restrict,No Action, Cascade,Set Null属性.现在分别对他们的属性含义做个解释. ON DELETE restrict(约束) ...