【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 铁人两项
题面 建立圆方树,考虑所有路径,发现路径上原来的点双(现在的方点)里的点都可以做中间点.但是路径上被方点夹着的圆点被计重了,要扣掉:枚举的两个端点也被算进去了,要扣掉.所以直接将方点权值设为点双大小, ...
随机推荐
- NO 18---- webpack 4.x 使用遇到的问题以及开发配置
最近在项目中用webpack的过程中老是出现问题,很是纳闷,按理说一直这样使用没有问题啊,经过我研究后发现,是因为在webpack更新到4.x之后,操作方式与之前相比变化很大.而我们使用npm默认安装 ...
- 关于Maven的一点理解
maven是一个项目管理工具,主要作用是: 1.依赖管理(jar包,工程之间); 2.统一开发规范和工具.完成项目的一步构建 3.工程聚合.继承.依赖 其核心配置文件就是pom.xml:pom即Pro ...
- 什么是mvc模式
MVC是一个架构,或者说是一个设计模式,它就是强制性使应用程序的输入,处理和输出分开.将一个应用程序分为三个部分:Model,View,Controller. 原理图: 分析: Model 模型(完成 ...
- Nginx笔记(一):安装
Nginx在安装前需要先安装其所依赖的类库,所以需先行安装好之后再进行Nginx安装. Nginx依赖以下模块: l gzip模块需要 zlib 库 l rewrite模块需要 pcre 库 l ...
- bootstrape3 模态
version 2 的模态 不能在 version 3 上面用. http://schy-hqh.iteye.com/blog/1995690
- tr命令详解
基础命令学习目录 原文链接:https://www.cnblogs.com/ginvip/p/6354440.html 什么是tr命令?tr,translate的简写,translate的翻译: [t ...
- 深入 JSX
从本质上讲,JSX 只是为 React.createElement(component, props, ...children) 函数提供的语法糖.JSX代码: 1 2 3 <MyButton ...
- Flask之笔记集合
目录 一.简述 二.基本使用 三.配置文件 四.路由系统 2.自定义正则路由 五.模版语言 六.请求和响应 七.Session 2.自定义session 八.蓝图 九.message 十.中间件 十一 ...
- Nginx是如何配置为 Web 服务器的【转载】
详解 Nginx是如何配置为 Web 服务器的 林涛 发表于:2016-11-29 23:23 分类:WebServer 标签:Nginx,web,web服务器 521次 抽象来说,将 Nginx 配 ...
- 【Alpha】Task分配与计划发布
团队项目链接 以上大概是我们的任务分配,根据目前的预计时间来看,到α版本项目稳定下来至少需要440小时的开发时间才能完成. 项目最大的问题点和难点在于其数据量非常之大,计算模块要求非常之多,想象一下 ...