[APIO2018]铁人两项 --- 圆方树
[APIO2018] 铁人两项
题目大意:
给定一张图,问有多少三元组(a,b,c)(a,b,c 互不相等)满足存在一条点不重复的以a为起点,经过b,终点为c的路径
如果你不会圆方树 ----------------------- 放弃是最好的选择(先学了再来吧)
如果你会圆方树
考虑\((a,...,c)\)
1.如果a,c不同属一个点双,不难发现答案为路上经过的(点双的节点个数)的和减去割点数
2.如果a,c同属一个点双,那么答案为本点双的节点个数 - 2
自然地想到方点的权值为内含节点个数(割点算在内),圆点权值为 -1
所以答案就是树上的路径权值之和了。
树形DP一下就好了 ------------------------------------------- 那就是错了。。。。
考虑的三元组中的a,c明显指圆点,方点是不能作为路径的起点和终点的。。。。。
于是,考虑每个点能出现的路径个数。
记\(sz[e]\)表示节点e在圆方树的子树中存在多少圆点
对于每个点,存在两种路径:


分两种情况统计即可
额外的,对于圆点,因为可能作为起点,终点,还要额外统计
莫名其妙的$loj$和$luogu$的$rk1$.......还是挺懵逼的
代码如下:
#include <cstdio>
#define sid 500050
using namespace std; long long ans;
int n, m, top, tim, nt, cnt, Asz, pp;
int dfn[sid], low[sid], st[sid], sz[sid], val[sid];
int cap[sid], acap[sid], nxt[sid], node[sid]; int read() { scanf("%d", &pp); return pp; }
void upmin(int &a, int b) { if(a > b) a = b; }
void Add(int u, int v) { nxt[++ cnt] = cap[u]; cap[u] = cnt; node[cnt] = v; }
void Bdd(int u, int v) { nxt[++ cnt] = acap[u]; acap[u] = cnt; node[cnt] = v; } void Tarjan(int e) {
dfn[e] = low[e] = ++ tim;
st[++ top] = e; sz[e] = ; val[e] = -;
for(int i = acap[e], d; i; i = nxt[i])
if(!dfn[d = node[i]]) {
Tarjan(d); upmin(low[e], low[d]);
if(low[d] < dfn[e]) continue;
int p; ++ nt; Add(e, nt);
do {
p = st[top --]; val[nt] ++;
Add(nt, p); sz[nt] += sz[p];
} while(p != d);
val[nt] ++; sz[e] += sz[nt];
}
else upmin(low[e], dfn[d]);
} void DP(int e) {
if(e <= n) ans += 1ll * (Asz - ) * val[e];
ans += 1ll * (Asz - sz[e]) * sz[e] * val[e];
for(int i = cap[e]; i; i = nxt[i]) {
int d = node[i];
DP(d), ans += 1ll * (Asz - sz[d]) * sz[d] * val[e];
}
} int main() {
nt = n = read(); m = read();
for(int i = ; i <= m; i ++) {
int u = read(), v = read();
Bdd(u, v); Bdd(v, u);
}
for(int i = ; i <= n; i ++)
if(!dfn[i])
Tarjan(i), Asz = sz[i], DP(i);
printf("%lld\n", ans);
return ;
}
[APIO2018]铁人两项 --- 圆方树的更多相关文章
- [APIO2018]铁人两项——圆方树+树形DP
题目链接: [APIO2018]铁人两项 对于点双连通分量有一个性质:在同一个点双里的三个点$a,b,c$,一定存在一条从$a$到$c$的路径经过$b$且经过的点只被经过一次. 那么我们建出原图的圆方 ...
- [APIO2018]铁人两项 [圆方树模板]
把这个图缩成圆方树,把方点的权值设成-1,圆点的权值设成点双的size,算 经过这个点的路径的数量*这个点的点权 的和即是答案. #include <iostream> #include ...
- [BZOJ5463][APIO2018]铁人两项(圆方树DP)
题意:给出一张图,求满足存在一条从u到v的长度大于3的简单路径的有序点对(u,v)个数. 做了上一题[HDU5739]Fantasia(点双连通分量+DP),这个题就是一个NOIP题了. 一开始考虑了 ...
- [APIO2018] Duathlon 铁人两项 圆方树,DP
[APIO2018] Duathlon 铁人两项 LG传送门 圆方树+简单DP. 不会圆方树的话可以看看我的另一篇文章. 考虑暴力怎么写,枚举两个点,答案加上两个点之间的点的个数. 看到题面中的一句话 ...
- 【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博 ...
- 洛谷P4630 铁人两项--圆方树
一道很好的圆方树入门题 感谢PinkRabbit巨佬的博客,讲的太好啦 首先是构建圆方树的代码,也比较好想好记 void tarjan(int u) { dfn[u] = low[u] = ++dfn ...
- [APIO2018]铁人两项(圆方树)
过了14个月再重新看这题,发现圆方树从来就没有写过.然后写了这题发现自己APIO2018打铁的原因竟然是没开long long,将树的部分的O(n)写挂了(爆int),毕竟去年APIO时我啥都不会,连 ...
随机推荐
- 解决ajax chrome禁止本地浏览时加载本地其他文件的方法
在chrome快捷键右键--属性 “ --allow-file-access-from-files ”,前面用空格隔开.然后应用--确定.
- HDU 2516 取石子游戏 (找规律)
题目链接 Problem Description 1堆石子有n个,两人轮流取.先取者第1次可以取任意多个,但不能全部取完.以后每次取的石子数不能超过上次取子数的2倍.取完者胜.先取者负输出" ...
- Go Web 编程 第一章 Web相关概念
第一章 Go与Web应用 Go学习群:415660935 1.1 Web应用 在计算机的世界里,应用(application)是一个与用户进行交互,并完成用户特定任务的软件程序.而Web应用则是部署在 ...
- Mac OS X 编译android内核 error: elf.h: No such file or directory 的解决方法
1. 从网上下个elf.h放到scripts/mod/文件夹(http://www.rockbox.org/tracker/9006?getfile=16683) 2. 修改两个文件mk_elfcon ...
- win10产品密钥 win10专业版激活码key
转载地址:http://www.xitongcheng.com/jiaocheng/xtazjc_article_35407.html https://blog.csdn.net/WangJianku ...
- oracle 一个网站
http://www.oracle.com/technetwork/cn/articles/11g-pivot-101924-zhs.html
- 一键去除网页BOM属性【解决乱码,头部空白,问题】
几个常出现的问题: 1.网站打开空白 2.页面头部出现多余的空白 3.网站出现乱码,如“锘�” 解决方法可以是: 1.选用专业的编辑器,例如notepad++,sublime,editplus这样不会 ...
- Codeforces 799B - T-shirt buying(STL)
题目链接:http://codeforces.com/problemset/problem/799/B 题目大意:有n件T恤,每件T体恤都分别有价格(每件衣服的价格不重复).前面的颜色.背部的颜色三种 ...
- window下线程同步之(Event Objects(事件))
Event 方式是最具弹性的同步机制,因为他的状态完全由你去决定,不会像 Mutex 和 Semaphores 的状态会由类似:WaitForSingleObject 一类的函数的调用而改变,所以你可 ...
- Python的hasattr() getattr() setattr() 函数使用方法(简介)
hasattr(object, name)判断一个对象里面是否有name属性或者name方法,返回BOOL值,有name特性返回True, 否则返回False.需要注意的是name要用括号括起来 1 ...