题目传送门

题目大意

  给定一棵树,问有多少个无序三元组$(x, y, z)$使得这三个不同点在树上两两距离相等。

  考虑这三个点构成的虚树。选取一个"舒适"的计数对象。

  其中黄色的点是关键点,绿色的点是虚树上的虚点。

  在树形动态规划的时候通常考虑一个点子树内的情况会比较简单。因此考虑将计数对象设为虚树上最浅的一个点。

性质1 最近公共祖先深度较深的两点到它们的最近公共祖先的距离相等。

  证明 设这两点为$A, B$,第三点为$C$。当三个点的最近公共祖先相同时显然。否则第三点到这两点距离不相等。

  当三个点的最近公共祖先不相同时。设$A$和$B$两点的最近公共祖先为$D$。

  因为三个点公共祖先不同所以$C$在$D$的子树外。  

  此时虚树上有5个点,因为$A$和$C$、$B$和$C$的最近公共祖先是相同点,设它为$E$。

  那么有$dis(A, D) + dis(D, E) + dis(E, C) = dis(B, D) + dis(D, E) + dis(E, C)$

  所以$dis(A, D) = dis(B, D)$。

  用$f[i][j]$表示在$i$的子树中,到点$i$距离为$j$的点的个数。

  如果点$x, y$的LCA的深度为$d$,且它们到它们的LCA的距离相等,$g[i][j]$表示在$i$的子树中,有多少个点对$(x, y)$使得它们的LCA到点$i$的距离为$d - j$。

  为了避免算重,应当边在合并子树的信息的时候边计算答案。

  除了$f[i][0]$初始为1,其他初始为0。

  设$i$的某个子节点为$k$。那么转移有(顺序不是这样的):

  $f[i][j] = f[i][j] + f[k][j - 1]$

  $g[i][j] = g[i][j] + g[k][j + 1]$

  $g[i][j] = g[i][j] + f[i][j] * f[k][j - 1]$ (当前点是较深的LCA)

  $result = result + f[i][j] * g[k][j - 1] + g[i][j] * f[k][j + 1]$

  于是状态$O(n^{2})$,时间复杂度也是满满的$O(n^{2})$。成功爆炸。

  注意到前两个转移只是挪一挪指针,而初始的时候$f, g$几乎可以看成没有值。因此这一部分考虑直接通过指针赋值来实现$O(1)$转移。

  问题是,这个只能做一次。那就选择深度最深的子树进行转移。

性质2 该种做法时间复杂度$O(n)$。

  证明 对树进行长链剖分。当且仅当一条边是虚边的时候需要暴力进行转移,时间复杂度是这棵子树的深度。

  这个深度的意义可以看成虚边深的一端连接的长链的长度。因为长链一直连向叶子结点,中间不会有其他虚边。

  又因为长链覆盖整棵树。因此时间复杂度$O(n)$。

  然后开一个内存池动态分配空间。(其实直接new也是可以的,只是慢一点)。  

Code

 /**
* bzoj
* Problem#3522 & 4543
* Accepted & Accepted
* Time: 44ms & 768ms
* Memory: 19652k & 21900k
*/
#include <iostream>
#include <cassert>
#include <cstdlib>
#include <cstdio>
#include <vector>
#ifndef WIN32
#define Auto "%lld"
#else
#define Auto "%I64d"
#endif
using namespace std;
typedef bool boolean;
#define ll long long const int N = ; ll pool[ * N];
ll* top = pool; ll* alloc(int len) {
ll* rt = top;
top += len;
return rt;
} int n;
ll res = ;
vector<int> mg[N];
int ml[N], longs[N];
ll *f[N], *g[N]; inline void init() {
scanf("%d", &n);
for (int i = , u, v; i < n; i++) {
scanf("%d%d", &u, &v);
mg[u].push_back(v);
mg[v].push_back(u);
}
} void dfs1(int p, int fa) {
ml[p] = ;
int maxl = -, id = -;
for (int i = ; i < (signed) mg[p].size(); i++) {
int e = mg[p][i];
if (e == fa) continue;
dfs1(e, p);
ml[p] = max(ml[p], ml[e] + );
if (ml[e] > maxl)
maxl = ml[e], id = e;
}
longs[p] = id;
} void dfs2(int p, int fa, int& maxlen, int blank) {
maxlen = max(maxlen, ml[p]);
if (longs[p] != -) {
dfs2(longs[p], p, maxlen, blank + );
res += g[longs[p]][];
f[p] = f[longs[p]] - , f[p][] = ;
g[p] = g[longs[p]] + ;
} else {
f[p] = alloc(maxlen + + blank) + blank;
g[p] = alloc(maxlen + + blank);
f[p][] = ;
} for (int i = ; i < (signed) mg[p].size(); i++) {
int e = mg[p][i], mxlen = ;
if (e == fa || e == longs[p]) continue;
dfs2(e, p, mxlen, );
// assert(f[p] + ml[e] + 1 < org[p] + siz[p]);
for (int j = ; j < ml[e]; j++)
res += f[p][j] * g[e][j + ];
for (int j = ; j <= ml[e] + ; j++)
res += g[p][j] * f[e][j - ];
for (int j = ; j <= ml[e] + ; j++)
g[p][j] += f[p][j] * f[e][j - ];
for (int j = ; j <= ml[e]; j++)
f[p][j + ] += f[e][j];
for (int j = ; j <= ml[e]; j++)
g[p][j - ] += g[e][j];
}
} inline void solve() {
int mxlen = ;
dfs1(, );
dfs2(, , mxlen, );
printf(Auto, res);
} int main() {
init();
solve();
return ;
}

bzoj 3522 / 4543 [POI 2014] Hotel - 动态规划 - 长链剖分的更多相关文章

  1. BZOJ.4543.[POI2014]Hotel加强版(长链剖分 树形DP)

    题目链接 弱化版:https://www.cnblogs.com/SovietPower/p/8663817.html. 令\(f[x][i]\)表示\(x\)的子树中深度为\(i\)的点的个数,\( ...

  2. BZOJ3522&4543 [POI2014]Hotel加强版 长链剖分

    上上周见fc爷用长链剖分秒题 于是偷偷学一学 3522的数据范围很小 可以暴力枚举每个点作为根节点来dp 复杂度$O(n^2)$ 考虑令$f[x][j]$表示以$x$为根的子树内距离$x$为$j$的点 ...

  3. bzoj4543 [POI2014]Hotel加强版 长链剖分+树形DP

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4543 题解 这道题的弱化版 bzoj3522 [POI2014]Hotel 的做法有好几种吧. ...

  4. 2018牛客网暑假ACM多校训练赛(第七场)I Tree Subset Diameter 动态规划 长链剖分 线段树

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round7-I.html 题目传送门 -  https://www.n ...

  5. BZOJ4543[POI2014]Hotel加强版——长链剖分+树形DP

    题意参见BZOJ3522 n<=100000 数据范围增强了,显然之前的转移方程不行了,那么不妨换一种. 因为不能枚举根来换根DP,那么我们描述的DP方程每个点要计算三个点都在这个点的子树内的方 ...

  6. 【BZOJ4543】[POI2014]Hotel加强版 长链剖分+DP

    [BZOJ4543][POI2014]Hotel加强版 Description 同OJ3522数据范围:n<=100000 Sample Input 7 1 2 5 7 2 5 2 3 5 6 ...

  7. BZOJ 4543 2016北京集训测试赛(二)Problem B: thr 既 长链剖分学习笔记

    Solution 这题的解法很妙啊... 考虑这三个点可能的形态: 令它们的重心为距离到这三个点都相同的节点, 则其中两个点分别在重心的两棵子树中, 且到重心的距离相等; 第三个点可能在重心的一棵不同 ...

  8. 【BZOJ4543】Hotel加强版(长链剖分)

    [BZOJ4543]Hotel加强版(长链剖分) 题面 BZOJ,没有题面 洛谷,只是普通版本 题解 原来我们的\(O(n^2)\)做法是设\(f[i][j]\)表示以\(i\)为根的子树中,距离\( ...

  9. BZOJ 3653: 谈笑风生(离线, 长链剖分, 后缀和)

    题意 给你一颗有 \(n\) 个点并且以 \(1\) 为根的树.共有 \(q\) 次询问,每次询问两个参数 \(p, k\) .询问有多少对点 \((p, a, b)\) 满足 \(p,a,b\) 为 ...

随机推荐

  1. gcc常用命令使用

    gcc编译文件过程 .c文件到 .i文件 到.s(汇编文件) 到.o文件,再到可执行文件 .c到.i 实操一下: test.c文件如下 : #include <stdlib.h> #inc ...

  2. vue路由(一个包含重定向、嵌套路由、懒加载的main.js如下)and 路由跳转传参的query和params的异同

    import Vue from 'vue'import VueRouter from 'vue-router'import App from './App'Vue.use(VueRouter)cons ...

  3. node.js初识07

    之前有说过,nodejs是 没有web容器的,阿帕奇是自带的web容器,如果希望node达到阿帕奇的效果,即http://127.0.0.1:3000/a/b/c.html 出现这样的链接访问页面,所 ...

  4. CSU 1849 Comparing answers(数学矩阵)

    Comparing answers 离散数学真的要好好学啊:一个邻接矩阵(这个矩阵一定是n×n的方阵,n是图的节点个数),表示的是从i到j有几条通路的时候,矩阵的1次方就代表从从i到j长度为1的路径通 ...

  5. 通过Referer设置来防盗链

    在Servlet中需要设置防盗链功能时可以通过以下代码: String referer = request.getHeader("Referer"); if(referer == ...

  6. EF切EFCore2.0存储过程问题

    在从EF切换成EFCore2.0的过程中,遇到了存储过程的实现问题. 在EF中调用存储过程,非常方便,能够直接将结果转换成对应的结果类. 如代码中的Database.SqlQuery<TElem ...

  7. 5.JVM的内存区域划分

    一.JVM介绍 1. 什么是JVM? JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟 ...

  8. Python -- Pandas介绍及简单实用【转】

    转http://www.datadependence.com/2016/05/scientific-python-pandas/ 一. Pandas简介 1.Python Data Analysis ...

  9. XML系列之--对电文格式XML的简单操作(三)

    前两章介绍了关于Linq创建.解析SOAP格式的XML,在实际运用中,可能会对xml进行一些其它的操作,比如基础的增删该查,而操作对象首先需要获取对象,针对于DOM操作来说,Linq确实方便了不少,如 ...

  10. 【Error】:svnrdump: E130003: The XML response contains invalid XML

    我这边的使用场景是在对远程服务器进行svnrdump dump操作时出现该问题,因为是对远程仓库多级子目录进行备份,结果出现错误. 在网上搜索了很多,有很多帖子是在checkout的时候出现问题,和我 ...