题目链接

题意大概是,求有多少三元组$(s,c,f)(s \neq c, c \neq f, s \neq f)$,满足从$s$到$f$有一条简单路径经过$c$。

得到结论:

  1. 点双中任意互不相同的三个点,必定存在一条简单路径依次经过这三个点。
  2. 显然,割点只能经过一次。

建出一棵圆方树,圆点的权值为$-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】铁人两项(圆方树)的更多相关文章

  1. [APIO2018] Duathlon 铁人两项 圆方树,DP

    [APIO2018] Duathlon 铁人两项 LG传送门 圆方树+简单DP. 不会圆方树的话可以看看我的另一篇文章. 考虑暴力怎么写,枚举两个点,答案加上两个点之间的点的个数. 看到题面中的一句话 ...

  2. [APIO2018]铁人两项 --- 圆方树

     [APIO2018] 铁人两项 题目大意: 给定一张图,问有多少三元组(a,b,c)(a,b,c 互不相等)满足存在一条点不重复的以a为起点,经过b,终点为c的路径 如果你不会圆方树 ------- ...

  3. [APIO2018]铁人两项 [圆方树模板]

    把这个图缩成圆方树,把方点的权值设成-1,圆点的权值设成点双的size,算 经过这个点的路径的数量*这个点的点权 的和即是答案. #include <iostream> #include ...

  4. [APIO2018]铁人两项——圆方树+树形DP

    题目链接: [APIO2018]铁人两项 对于点双连通分量有一个性质:在同一个点双里的三个点$a,b,c$,一定存在一条从$a$到$c$的路径经过$b$且经过的点只被经过一次. 那么我们建出原图的圆方 ...

  5. 洛谷P4630 铁人两项--圆方树

    一道很好的圆方树入门题 感谢PinkRabbit巨佬的博客,讲的太好啦 首先是构建圆方树的代码,也比较好想好记 void tarjan(int u) { dfn[u] = low[u] = ++dfn ...

  6. 【Luogu4630】【APIO2018】 Duathlon 铁人两项 (圆方树)

    Description ​ 给你一张\(~n~\)个点\(~m~\)条边的无向图,求有多少个三元组\(~(x, ~y, ~z)~\)满足存在一条从\(~x~\)到\(~z~\)并且经过\(~y~\)的 ...

  7. LOJ 2587 「APIO2018」铁人两项——圆方树

    题目:https://loj.ac/problem/2587 先写了 47 分暴力. 对于 n<=50 的部分, n3 枚举三个点,把图的圆方树建出来,合法条件是 c 是 s -> f 路 ...

  8. loj2587 「APIO2018」铁人两项[圆方树+树形DP]

    主要卡在一个结论上..关于点双有一个常用结论,也经常作为在圆方树/简单路径上的良好性质,对于任意点双内互不相同的三点$s,c,t$,都存在简单路径$s\to c\to t$,证明不会.可以参见clz博 ...

  9. [BZOJ5463][APIO2018]铁人两项(圆方树DP)

    题意:给出一张图,求满足存在一条从u到v的长度大于3的简单路径的有序点对(u,v)个数. 做了上一题[HDU5739]Fantasia(点双连通分量+DP),这个题就是一个NOIP题了. 一开始考虑了 ...

  10. 解题:APIO 2018 铁人两项

    题面 建立圆方树,考虑所有路径,发现路径上原来的点双(现在的方点)里的点都可以做中间点.但是路径上被方点夹着的圆点被计重了,要扣掉:枚举的两个端点也被算进去了,要扣掉.所以直接将方点权值设为点双大小, ...

随机推荐

  1. Haproxy介绍、安装与配置

    Haproxy技术详解一. 介绍HAProxy是一个使用C语言编写的自由及开放源代码软件,其提供高可用性.负载均衡,以及基于TCP和HTTP的应用程序代理.HAProxy特别适用于那些负载特大的web ...

  2. jsp标签在JavaScript中使用时,可能会出现的一个问题。

    直接上代码 <script type="text/javascript"> var E = window.wangEditor; var editor = new E( ...

  3. 多表查询sql语句

    多表查询sql语句 1 --解锁SCOTT用户 2 alter user scott account unlock 3 --检索指定的列 4 select job,ename,empno from e ...

  4. CHAPTER 38 Reading ‘the Book of Life’ The Human Genome Project 第38章 阅读生命之书 人体基因组计划

    CHAPTER 38 Reading ‘the Book of Life’ The Human Genome Project 第38章 阅读生命之书 人体基因组计划 Humans have about ...

  5. Netty源码分析第5章(ByteBuf)---->第10节: SocketChannel读取数据过程

    Netty源码分析第五章: ByteBuf 第十节: SocketChannel读取数据过程 我们第三章分析过客户端接入的流程, 这一小节带大家剖析客户端发送数据, Server读取数据的流程: 首先 ...

  6. Flink 部署文档

    Flink 部署文档 1 先决条件 2 下载 Flink 二进制文件 3 配置 Flink 3.1 flink-conf.yaml 3.2 slaves 4 将配置好的 Flink 分发到其他节点 5 ...

  7. R软件中 文本分析安装包 Rjava 和 Rwordseg 傻瓜式安装方法四部曲

    这两天,由于要做一个文本分析的内容,所以搜索了一天R语言中的可以做文本分析的加载包,但是在安装包的过程,真是被虐千百遍,总是安装不成功.特此专门写一篇博文,把整个心塞史畅快的释放一下. ------- ...

  8. Python字典 (dict)

    作者博文地址:http://www.cnblogs.com/spiritman/ 字典是Python语言中唯一的映射类型.字典对象是可变的,它是一个容器类型,支持异构.任意嵌套. 创建字典 语法:{k ...

  9. 凡事不求甚解,遇事必定抓瞎——PHP开发Apache服务器配置备忘录

    照此配置流程,绝对一路畅通,可保无虞. 昨天弄了个PHP小程序,想在本地跑一下测试,可是工作电脑没有安装环境,于是下载了一个wamp,一路畅通,Apache.Mysql.PHP就全有了.启动wamp服 ...

  10. CF 1064B Equations of Mathematical Magic(思维规律)

    Description Colossal! — exclaimed Hawk-nose. — A programmer! That's exactly what we are looking for. ...