bzoj 3522 / 4543 [POI 2014] Hotel - 动态规划 - 长链剖分
题目传送门
题目大意
给定一棵树,问有多少个无序三元组$(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 - 动态规划 - 长链剖分的更多相关文章
- BZOJ.4543.[POI2014]Hotel加强版(长链剖分 树形DP)
题目链接 弱化版:https://www.cnblogs.com/SovietPower/p/8663817.html. 令\(f[x][i]\)表示\(x\)的子树中深度为\(i\)的点的个数,\( ...
- BZOJ3522&4543 [POI2014]Hotel加强版 长链剖分
上上周见fc爷用长链剖分秒题 于是偷偷学一学 3522的数据范围很小 可以暴力枚举每个点作为根节点来dp 复杂度$O(n^2)$ 考虑令$f[x][j]$表示以$x$为根的子树内距离$x$为$j$的点 ...
- bzoj4543 [POI2014]Hotel加强版 长链剖分+树形DP
题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4543 题解 这道题的弱化版 bzoj3522 [POI2014]Hotel 的做法有好几种吧. ...
- 2018牛客网暑假ACM多校训练赛(第七场)I Tree Subset Diameter 动态规划 长链剖分 线段树
原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round7-I.html 题目传送门 - https://www.n ...
- BZOJ4543[POI2014]Hotel加强版——长链剖分+树形DP
题意参见BZOJ3522 n<=100000 数据范围增强了,显然之前的转移方程不行了,那么不妨换一种. 因为不能枚举根来换根DP,那么我们描述的DP方程每个点要计算三个点都在这个点的子树内的方 ...
- 【BZOJ4543】[POI2014]Hotel加强版 长链剖分+DP
[BZOJ4543][POI2014]Hotel加强版 Description 同OJ3522数据范围:n<=100000 Sample Input 7 1 2 5 7 2 5 2 3 5 6 ...
- BZOJ 4543 2016北京集训测试赛(二)Problem B: thr 既 长链剖分学习笔记
Solution 这题的解法很妙啊... 考虑这三个点可能的形态: 令它们的重心为距离到这三个点都相同的节点, 则其中两个点分别在重心的两棵子树中, 且到重心的距离相等; 第三个点可能在重心的一棵不同 ...
- 【BZOJ4543】Hotel加强版(长链剖分)
[BZOJ4543]Hotel加强版(长链剖分) 题面 BZOJ,没有题面 洛谷,只是普通版本 题解 原来我们的\(O(n^2)\)做法是设\(f[i][j]\)表示以\(i\)为根的子树中,距离\( ...
- BZOJ 3653: 谈笑风生(离线, 长链剖分, 后缀和)
题意 给你一颗有 \(n\) 个点并且以 \(1\) 为根的树.共有 \(q\) 次询问,每次询问两个参数 \(p, k\) .询问有多少对点 \((p, a, b)\) 满足 \(p,a,b\) 为 ...
随机推荐
- pyqt常用窗口组件
扩展知识: 熟悉常用的窗口组件: 1 按钮类 QPushButton 普通按钮 QToolButton 工具按钮:通常在工具栏使用 QRadioButton 单选框 QCheckBox ...
- Navicat Premium 安装与激活破解版简单操作 (原)
首先下载navicate程序以及破解文件,这里一并存到了百度网盘直接下载即可 链接:https://pan.baidu.com/s/11ptFmsV1o3B5oB00zm2NdQ 密码:yw82 解压 ...
- QT 添加 lib库
扒自网友文章: 一.添加第三方的头文件 首先,添加头文件 #include "ControlCAN.h" 然后,再将这个头文件放到工程的目录下 二.添加.lib文件 首先,将.l ...
- unity之复制文本到剪贴板
代码如下: static void CopyString(string str) { TextEditor te = new TextEditor(); te.text = str; te.Selec ...
- arm trustzone
arm的trustzone并不涉及到具体的crypto算法,只是实现: 1) 敏感信息的安全存储: 2) 控制bus和debug的安全访问,保证信息不被泄露: trustzone是system_lev ...
- 《大话设计模式》c++实现 工厂模式
工厂模式 工厂模式(Factory Pattern)是 Java 中最常用的设计模式之一.这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式. 在工厂模式中,我们在创建对象时不会对客户端 ...
- Python全栈-day15-day16-常用模块
1.time模块 1)时间戳 import time# 通常来说,时间戳表示的是从1970年1月1日00:00:00开始按秒计算的偏移量 # 偏移量的是float类型 start_time = tim ...
- jQuery事件--blur()和focus()
blur([[data],fn]) 概述 当元素失去焦点时触发 blur 事件. 这个函数会调用执行绑定到blur事件的所有函数,包括浏览器的默认行为.可以通过返回false来防止触发浏览器的默 ...
- 基于KVM、Xen、OpenVZ等虚拟化技术的WEB在线管理工具
1.Proxmox proxmox是一个开源的虚拟化管理平台,支持集群管理和HA.在存储方面,proxmox除了支持常用的lvm,nfs,iscsi,还支持集群存储glusterfs和ceph,这也是 ...
- quick-cocos2d-x 游戏开发之一】开发工具sublime text及其强力插件QuickXDev
http://blog.csdn.net/yihaiyiren/article/details/17220927