传送门


要求的是一条按顺序经过\(s,t,c\)三个点的简单路径。简单路径的计数问题不难想到点双联通分量,进而使用圆方树进行求解。

首先将原图缩点,对于一个大小为\(size\)的点双联通分量内,在这个分量内部任意选择\(s,t,c\)都是可行的,可以贡献\(P_{size}^3\)的答案。

接下来就需要计算跨越点双联通分量的\(s,t,c\)了。这个可以在圆方树上进行树形DP统计答案。

但是考虑到割点可能会被统计多次,我们令圆方树中所有的方点都不包含它的父亲,这样对于一个圆点就只会被计入一次答案了。当然了,为了不重不漏地统计每一种可行方案,上面的一系列的贡献都会有一些变化。

具体是这样子的:

首先缩点,对于一个点双联通分量,只贡献\(P_{size-1}^3 + P_{size-1}^2\)的答案,因为从去掉方点父亲的点双联通分量中拿出两个点作为\(s,t\),将方点父亲作为\(c\)的情况可以在树形DP的过程中统计到

接着在圆方树上树形DP。设\(f_x\)表示\(x\)及其子树中取一个点作为\(s\)的方案数(其实就是子树中的圆点个数),\(g_x\)表示\(x\)及其子树中取两个点作为\(s,t\)的方案数

注意到取一个点作为\(s\)与取一个点作为\(c\)是等价的,所以DP贡献的答案中需要乘上2

在合并过程中,对于一个圆点,它对\(f\)和\(g\)的贡献放在它的父亲方点上进行。

对于一个圆点\(x\),它的孩子方点\(ch\)到它的转移为:

\(ans += 2 \times ( f_x \times g_{ch} + f_{ch} \times g_x + f_x \times f_{ch})\)

\(f_x += f_{ch}\)

\(g_x += g_{ch}\)

额外乘的\(f_x \times f_{ch}\)表示以当前这个圆点作为\(t\)的方案数

对于一个方点\(y\),先合并它的孩子节点\(ch\):

\(ans += 2 \times ( f_y \times g_{ch} + f_{ch} \times g_y + f_y \times f_{ch} \times size_y)\)

\(f_y += f_{ch}\)

\(g_y += g_{ch}\)

其中\(size_y\)表示\(y\)对应的点双联通分量的大小

然后考虑当前这个点双联通分量的贡献:

\(ans += 2 \times (f_x \times P_{size_y - 1} ^ 2 + g_y \times (size_y - 1))\)

\(f_y += size_y - 1\)

\(g_y += P_{size_y - 1}^2\)

最后对于树根\(root\)并没有计算贡献,最后答案加上\(2 \times g_{root}\)即可。

注意图可能不连通

#include<bits/stdc++.h>
//This code is written by Itst
using namespace std; inline int read(){
int a = 0;
char c = getchar();
bool f = 0;
while(!isdigit(c) && c != EOF){
if(c == '-')
f = 1;
c = getchar();
}
if(c == EOF)
exit(0);
while(isdigit(c)){
a = a * 10 + c - 48;
c = getchar();
}
return f ? -a : a;
} const int MAXN = 2e5 + 7;
struct Edge{
int end , upEd;
}Ed[MAXN << 1];
int head[MAXN] , N , M , cnt , cntEd;
int dfn[MAXN] , low[MAXN] , st[MAXN] , top , ts;
long long ans , dp1[MAXN] , dp2[MAXN];
vector < int > ch[MAXN]; inline void addEd(int a , int b){
Ed[++cntEd].end = b;
Ed[cntEd].upEd = head[a];
head[a] = cntEd;
} void pop(int t , int bot){
++cnt;
ch[t].push_back(cnt);
long long c = 0;
do{
++c;
ch[cnt].push_back(st[top]);
}while(st[top--] != bot);
ans += c * (c - 1) / 2 + c * (c - 1) * (c - 2) / 2;
} void tarjan(int x , int p){
dfn[x] = low[x] = ++ts;
st[++top] = x;
for(int i = head[x] ; i ; i = Ed[i].upEd)
if(Ed[i].end != p)
if(!dfn[Ed[i].end]){
tarjan(Ed[i].end , x);
low[x] = min(low[x] , low[Ed[i].end]);
if(low[Ed[i].end] >= dfn[x])
pop(x , Ed[i].end);
}
else
low[x] = min(low[x] , dfn[Ed[i].end]);
} void dfs(int x){
for(int i = 0 ; i < ch[x].size() ; ++i){
dfs(ch[x][i]);
ans += dp1[x] * dp2[ch[x][i]] + dp1[ch[x][i]] * dp2[x] + dp1[x] * dp1[ch[x][i]] * (x <= N ? 1 : ch[x].size() + 1);
dp2[x] += dp2[ch[x][i]];
dp1[x] += dp1[ch[x][i]];
}
if(x > N){
ans += dp1[x] * ch[x].size() * (ch[x].size() - 1) + dp2[x] * ch[x].size();
dp2[x] = dp2[x] + dp1[x] * ch[x].size() + 1ll * ch[x].size() * (ch[x].size() - 1);
dp1[x] += ch[x].size();
}
} int main(){
#ifndef ONLINE_JUDGE
freopen("in","r",stdin);
freopen("out","w",stdout);
#endif
N = cnt = read();
M = read();
for(int i = 1 ; i <= M ; ++i){
int a = read() , b = read();
addEd(a , b);
addEd(b , a);
}
for(int i = 1 ; i <= N ; ++i)
if(!dfn[i]){
top = 0;
tarjan(i , 0);
dfs(i);
ans += dp2[i];
}
cout << ans * 2;
return 0;
}

Luogu4630 APIO2018 Duathlon 圆方树、树形DP的更多相关文章

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

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

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

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

  3. 2019.03.29 bzoj5463: [APIO2018] 铁人两项(圆方树+树形dp)

    传送门 题意简述:给你一张无向图,问你满足存在从a−>b−>ca->b->ca−>b−>c且不经过重复节点的路径的有序点对(a,b,c)(a,b,c)(a,b,c) ...

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

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

  5. LOJ.2587.[APIO2018]铁人两项Duathlon(圆方树)

    题目链接 LOJ 洛谷P4630 先对这张图建圆方树. 对于S->T这条(些)路径,其对答案的贡献为可能经过的所有点数,那么我们把方点权值设为联通分量的大小,可以直接去求树上路径权值和. 因为两 ...

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

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

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

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

  8. 【NOI P模拟赛】仙人掌(圆方树,树形DP)

    题面 n n n 个点, m m m 条边. 1 ≤ n ≤ 1 0 5 , n − 1 ≤ m ≤ 2 × 1 0 5 1\leq n\leq 10^5,n-1\leq m\leq 2\times1 ...

  9. 洛谷P4630 [APIO2018] Duathlon 铁人两项 【圆方树】

    题目链接 洛谷P4630 题解 看了一下部分分,觉得树的部分很可做,就相当于求一个点对路径长之和的东西,考虑一下能不能转化到一般图来? 一般图要转为树,就使用圆方树呗 思考一下发现,两点之间经过的点双 ...

随机推荐

  1. video 铺满父元素(object-fit: fill;)

    遇到这个属性,是在给video 嵌入一个div时,导致video播放器上下有灰色.在控制台查看video默认样式的时候看到了这个属性. 播放器上下的灰色,不是我们想要的样式,如果能完全覆盖就更好了. ...

  2. mysql 优化配置参数(my.cnf)

    max_connections:允许客户端并发连接的最大数量,默认值是151,一般将该参数设置为500-2000max_connect_errors:如果客户端尝试连接的错误数量超过这个参数设置的值, ...

  3. android 事件反拦截

    有一种方法可以阻止父层的View截获touch事件,就是调用 getParent().requestDisallowInterceptTouchEvent(true);方法.一旦底层View收到tou ...

  4. scrapy系列(四)——CrawlSpider解析

    CrawlSpider也继承自Spider,所以具备它的所有特性,这些特性上章已经讲过了,就再在赘述了,这章就讲点它本身所独有的. 参与过网站后台开发的应该会知道,网站的url都是有一定规则的.像dj ...

  5. 第四周读书笔记——读《我是一只IT小小鸟》有感

             读<我是一只IT小小鸟>有感 这是邓老师倾力推荐的一本书.这本书的标题化用了我们耳熟能详的歌词,算是较有新意吧.更重点在于,这本书的作者不是哪一位大牛,而是一群刚刚走出校 ...

  6. SqlServer索引页损坏恢复

    问题背景 运维操作失误,在没有正常关闭sqlserver的情况下,将服务器关闭了,重启后某些表损坏(应该是某些页损坏了,没有损坏的页还能访问到数据,但是访问损坏了的页就有问题),目前数据库只有4.20 ...

  7. webApi core2 DI通过代码来获取容器里面已注入的对象

    请求服务 来自 HttpContext 的一次 ASP.NET 请求中可用的服务通过 RequestServices 集合公开的. 请求服务将你配置的服务和请求描述为应用程序的一部分.当你的对象指定依 ...

  8. Linux下内存查看命令

    在Linux下面,我们常用top命令来查看系统进程,top也能显示系统内存.我们常用的Linux下查看内容的专用工具是free命令. Linux下内存查看命令free详解: 在Linux下查看内存我们 ...

  9. centos 6.5下安装nmap工具及简单用法

    Nmap是一款针对大型网络的端口扫描工具,被广泛应用于黑客领域做漏洞探测以及安全扫描,其主要功能有主机发现(Host Discovery). 端口扫描(Port Scanning). 版本侦测(Ver ...

  10. Kubernetes的搭建与配置(一):集群环境搭建

    1.环境介绍及准备: 1.1 物理机操作系统 物理机操作系统采用Centos7.3 64位,细节如下. [root@localhost ~]# uname -a Linux localhost.loc ...