题目传送门

题目大意

  给定一棵树,问有多少个无序三元组$(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. iOS 新浪微博-2.0 搜索框/标题带箭头/下拉菜单

    不管是搜索框还是下拉菜单,我们都需要对背景的图片进行拉伸.定义一个Category分类对图片进行操作. UIImage+Effect.h #import <UIKit/UIKit.h> @ ...

  2. 回车和刷新以及Ctr+F5的区别

    回车(url跳转)主要是判断本地缓存文件的Expires的有效时间,如果有效则直接使用客户端缓存 不在提交到HTTP服务器 F5 Expires设置不再起效果,只有Last-Modified/ETag ...

  3. 一个站点配置多个App.config

    一个项目一般都只有一个配置文件.web项目中用的是web.config,但项目中有时候需要单独来配置一个文件.比如:app.config,那是否可以呢? 答案是可以的.可以在web.config中指定 ...

  4. gitlab4.0_安装

    一,安装环境 OS:redhat7.4 二,安装依赖包 yum -y groupinstall 'Development Tools'  ===>待验证 yum -y install pytho ...

  5. MyBatis基础入门《十八》动态SQL(if-where)

    MyBatis基础入门<十八>动态SQL(if-where) 描述: 代码是在<MyBatis基础入门<十七>动态SQL>基础上进行改造的,不再贴所有代码,仅贴改动 ...

  6. JavaScript-判断指定日期是一年中第几天-按照从大到小的顺序输出

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...

  7. sqlserver中查询表字段的sql语句

    sqlserver中的表信息字段信息这些东西也是放到系统表中的,以下sql语句用于查询某表的字段信息. select t1.id object_id,t1.name object_name,t2.va ...

  8. python SQLite说一点点, python使用数据库需要注意的几点

    SQLite是一种嵌入式数据库,它的数据库就是一个文件.由于SQLite本身是C写的,而且体积很小,所以,经常被集成到各种应用程序中,甚至在iOS和Android的App中都可以集成. Python就 ...

  9. 20165305 苏振龙《Java程序设计》第四周学习总结

    第五章 继承: 面向对象中,为避免多个类间重复定义共同行为.(简单说就是将相同的程序代码提升为父类.) 特点: 这里接触到了新的关键词,extends,在java语言中用estends来继承父类的行为 ...

  10. Class__One HomeWork 实验报告

    石家庄铁道大学信息科学与技术学院       实验报告 2018年----2019年  第一学期               题目:   四则运算和验证码 课程名称:  JAVA语言程序设计 班    ...