【APIO 2018】铁人两项(圆方树)
题意大概是,求有多少三元组$(s,c,f)(s \neq c, c \neq f, s \neq f)$,满足从$s$到$f$有一条简单路径经过$c$。
得到结论:
- 点双中任意互不相同的三个点,必定存在一条简单路径依次经过这三个点。
- 显然,割点只能经过一次。
建出一棵圆方树,圆点的权值为$-1$,方点的权值为该点双中点的个数,那任意两个圆点之间可以作为它们中转点的个数就是它们在圆方树上路径的点权和。
具体来讲就是割点上只能经过一次,圆点设成$-1$是为了去重方便。
以前只写过点双缩树,这里写圆方树更方便,权且将这道题作为学习的例题吧。
建圆方树只要在Tarjan上稍作修改,这里给出建树的例子:
void Tarjan(int x, int fa) {
sta[++top] = x; in[x] = -;
dfn[x] = low[x] = ++_clock;
for (int i = las[x]; i; i = pre[i]) {
if (to[i] == fa) continue;
if (dfn[to[i]]) {
low[x] = std::min(low[x], dfn[to[i]]);
} else {
Tarjan(to[i], x);
low[x] = std::min(low[x], low[to[i]]);
if (dfn[x] <= low[to[i]]) {
xtr[x].push_back(++c_n); ++in[c_n];
for (int t = -; t != to[i]; ) {
t = sta[top--];
xtr[c_n].push_back(t); ++in[c_n];
}
}
}
}
}
(注:xtr为圆方树,in < 0代表圆点,否则代表了该方点中点双里点的个数)
之后这个题就好做了,统计的是所有圆点点对间路径长度的总和。
#include <cstdio>
#include <vector>
#include <iostream> typedef long long LL;
const int N = ; int n, m;
int dfn[N], low[N], in[N], siz[N], sta[N], top, _clock, c_n;
std::vector<int> xtr[N];
LL ans, sum[N]; int yun = , las[N], to[N << ], pre[N << ];
inline void Add(int a, int b) {
to[++yun] = b; pre[yun] = las[a]; las[a] = yun;
} void Tarjan(int x, int fa) {
sta[++top] = x; in[x] = -;
dfn[x] = low[x] = ++_clock;
for (int i = las[x]; i; i = pre[i]) {
if (to[i] == fa) continue;
if (dfn[to[i]]) {
low[x] = std::min(low[x], dfn[to[i]]);
} else {
Tarjan(to[i], x);
low[x] = std::min(low[x], low[to[i]]);
if (dfn[x] <= low[to[i]]) {
xtr[x].push_back(++c_n); ++in[c_n];
for (int t = -; t != to[i]; ) {
t = sta[top--];
xtr[c_n].push_back(t); ++in[c_n];
}
}
}
}
} void Dfs(int x) {
if (in[x] < ) siz[x] = , sum[x] = in[x];
for (int i = ; i < (int)xtr[x].size(); ++i) {
int v = xtr[x][i];
Dfs(v);
siz[x] += siz[v];
sum[x] += (LL) siz[v] * in[x] + sum[v];
}
}
void Calc(int x) {
if (in[x] < ) ans += sum[x] - in[x];
LL cnt = ;
for (int i = ; i < (int)xtr[x].size(); ++i) {
int v = xtr[x][i];
ans += (LL) sum[v] * (siz[x] - (in[x] < ) - siz[v]);
cnt += (LL) siz[v] * (siz[x] - (in[x] < ) - siz[v]);
}
ans += (LL) cnt / * in[x];
for (int i = ; i < (int)xtr[x].size(); ++i) {
int v = xtr[x][i];
Calc(v);
}
} int main() {
scanf("%d%d", &n, &m);
c_n = n;
for (int i = , x, y; i <= m; ++i) {
scanf("%d%d", &x, &y);
Add(x, y); Add(y, x);
}
for (int i = ; i <= n; ++i) {
if (!dfn[i]) {
Tarjan(i, );
Dfs(i);
Calc(i);
}
}
printf("%lld\n", ans * ); return ;
}
$\bigodot$技巧&套路:
- 圆方树的构建,圆方树上的统计技巧,可以用圆点的权值设成$-1$来去重。
【APIO 2018】铁人两项(圆方树)的更多相关文章
- [APIO2018] Duathlon 铁人两项 圆方树,DP
[APIO2018] Duathlon 铁人两项 LG传送门 圆方树+简单DP. 不会圆方树的话可以看看我的另一篇文章. 考虑暴力怎么写,枚举两个点,答案加上两个点之间的点的个数. 看到题面中的一句话 ...
- [APIO2018]铁人两项 --- 圆方树
[APIO2018] 铁人两项 题目大意: 给定一张图,问有多少三元组(a,b,c)(a,b,c 互不相等)满足存在一条点不重复的以a为起点,经过b,终点为c的路径 如果你不会圆方树 ------- ...
- [APIO2018]铁人两项 [圆方树模板]
把这个图缩成圆方树,把方点的权值设成-1,圆点的权值设成点双的size,算 经过这个点的路径的数量*这个点的点权 的和即是答案. #include <iostream> #include ...
- [APIO2018]铁人两项——圆方树+树形DP
题目链接: [APIO2018]铁人两项 对于点双连通分量有一个性质:在同一个点双里的三个点$a,b,c$,一定存在一条从$a$到$c$的路径经过$b$且经过的点只被经过一次. 那么我们建出原图的圆方 ...
- 洛谷P4630 铁人两项--圆方树
一道很好的圆方树入门题 感谢PinkRabbit巨佬的博客,讲的太好啦 首先是构建圆方树的代码,也比较好想好记 void tarjan(int u) { dfn[u] = low[u] = ++dfn ...
- 【Luogu4630】【APIO2018】 Duathlon 铁人两项 (圆方树)
Description 给你一张\(~n~\)个点\(~m~\)条边的无向图,求有多少个三元组\(~(x, ~y, ~z)~\)满足存在一条从\(~x~\)到\(~z~\)并且经过\(~y~\)的 ...
- LOJ 2587 「APIO2018」铁人两项——圆方树
题目:https://loj.ac/problem/2587 先写了 47 分暴力. 对于 n<=50 的部分, n3 枚举三个点,把图的圆方树建出来,合法条件是 c 是 s -> f 路 ...
- loj2587 「APIO2018」铁人两项[圆方树+树形DP]
主要卡在一个结论上..关于点双有一个常用结论,也经常作为在圆方树/简单路径上的良好性质,对于任意点双内互不相同的三点$s,c,t$,都存在简单路径$s\to c\to t$,证明不会.可以参见clz博 ...
- [BZOJ5463][APIO2018]铁人两项(圆方树DP)
题意:给出一张图,求满足存在一条从u到v的长度大于3的简单路径的有序点对(u,v)个数. 做了上一题[HDU5739]Fantasia(点双连通分量+DP),这个题就是一个NOIP题了. 一开始考虑了 ...
- 解题:APIO 2018 铁人两项
题面 建立圆方树,考虑所有路径,发现路径上原来的点双(现在的方点)里的点都可以做中间点.但是路径上被方点夹着的圆点被计重了,要扣掉:枚举的两个端点也被算进去了,要扣掉.所以直接将方点权值设为点双大小, ...
随机推荐
- 使用proxyee-down解决百度云下载限速问题
1.在下面页面安装HTTP下载器 https://github.com/proxyee-down-org/proxyee-down#%E4%B8%8B%E8%BD%BD 2.安装switchy插件 h ...
- js数组的比较
如果两个数组元素个数都相等,但排序不同,那么它两个相等吗?结果肯定是否定的.但如果先调用sort()方法进行排序,结果就是true了. console.log(a.sort().toString()= ...
- LeetCode 192. Word Frequency
分析 写bash,不太会啊…… 难度 中 来源 https://leetcode.com/problems/word-frequency/ 题目 Write a bash script to calc ...
- if _ else if _ else,case,程序逻辑判断- java基础
//单个判端 if(){ } //双判端 if(){ }else{ } //多重判端 if(){ }else if(){ }else if(){ }else{ } package test1; // ...
- Linux系统下搭建FTP/SFTP服务器
传输文件经常使用ftp和sftp服务器.Windows下有多种可视化工具,使用快捷.Linux经常需要自行搭建这两种服务器,当然搭建熟练的话,会更加快捷. 1.检查Linux系统是否安装了vsftp和 ...
- linux安装nginx并配置负载均衡
linux上安装nginx比较简单: 前提是需要有gcc或者g++ 1.yum需要的依赖 yum -y install openssl openssl-devel 2.解压pcre库.zlib库 ...
- SpringBoot初始教程之Redis集中式Session管理
1.介绍 有关Session的管理方式这里就不再进行讨论,目前无非就是三种单机Session(基于单机内存,无法部署多台机器).基于Cookie(安全性差).基于全局的统一Session管理(redi ...
- 20162328蔡文琛 week05 大二
20162328 2017-2018-1 <程序设计与数据结构>第5周学习总结 教材学习内容总结 集合是收集元素并组织其他对象的对象. 集合中的元素一般由加入集合的次序或元素之间的某些固有 ...
- 08慕课网《进击Node.js基础(一)》事件events
引用events模块中的EventEmitter 事件的监听和发射 相同的事件发射数量有限,可以通过setMaxListeners设置峰值 var EventEmitter = require('ev ...
- Sprint10
进展:设置事件提醒部分已经完成,接下来是实现完成后在添加主界面显示已添加的事件及时间,并可设置可用与不可用.