[APIO2018]铁人两项——圆方树+树形DP
题目链接:
对于点双连通分量有一个性质:在同一个点双里的三个点$a,b,c$,一定存在一条从$a$到$c$的路径经过$b$且经过的点只被经过一次。
那么我们建出原图的圆方树,枚举中间点$b$,一对合法的$a,c$需要使这两个点位于与$b$直接相连的方点的不同子树中。树形$DP$,对圆点和方点分别统计答案即可。
#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<vector>
#include<bitset>
#include<cstring>
#include<iostream>
#include<algorithm>
#define ll long long
using namespace std;
int n,m;
int x,y;
vector<int>q[200010];
int head[100010];
int to[400010];
int next[400010];
int size[200010];
int low[100010];
int dfn[100010];
int tot;
int cnt;
int num;
int st[100010];
int top;
int vis[200010];
ll ans;
int sum;
void add(int x,int y)
{
tot++;
next[tot]=head[x];
head[x]=tot;
to[tot]=y;
}
void insert(int x,int y)
{
q[x].push_back(y);
}
void tarjan(int x)
{
st[++top]=x;
low[x]=dfn[x]=++num;
for(int i=head[x];i;i=next[i])
{
if(!dfn[to[i]])
{
tarjan(to[i]);
low[x]=min(low[x],low[to[i]]);
if(low[to[i]]>=dfn[x])
{
insert(++cnt,x);
insert(x,cnt);
int now=0;
do
{
now=st[top--];
insert(cnt,now);
insert(now,cnt);
}
while(now!=to[i]);
}
}
else
{
low[x]=min(low[x],dfn[to[i]]);
}
}
}
void dfs(int x,int fa)
{
vis[x]=1;
size[x]=(x<=n?1:0);
int len=q[x].size();
for(int i=0;i<len;i++)
{
int to=q[x][i];
if(to!=fa)
{
dfs(to,x);
size[x]+=size[to];
}
}
}
void tree_dp(int x,int fa)
{
int len=q[x].size();
for(int i=0;i<len;i++)
{
int to=q[x][i];
if(to!=fa)
{
tree_dp(to,x);
}
}
if(x<=n)
{
for(int i=0;i<len;i++)
{
int to=q[x][i];
if(to!=fa)
{
ans+=1ll*size[to]*(sum-size[to]-1);
}
else
{
ans+=1ll*(sum-size[x])*(size[x]-1);
}
}
}
else if(len>=3)
{
for(int i=0;i<len;i++)
{
int to=q[x][i];
if(to!=fa)
{
ans+=1ll*size[to]*(sum-size[to])*(len-2);
}
else
{
ans+=1ll*(sum-size[x])*size[x]*(len-2);
}
}
}
}
int main()
{
scanf("%d%d",&n,&m);cnt=n;
for(int i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
add(x,y);
add(y,x);
}
for(int i=1;i<=n;i++)
{
if(!dfn[i])
{
tarjan(i);
}
}
for(int i=1;i<=cnt;i++)
{
if(!vis[i])
{
dfs(i,0);
sum=size[i];
tree_dp(i,0);
}
}
printf("%lld",ans);
}
[APIO2018]铁人两项——圆方树+树形DP的更多相关文章
- loj2587 「APIO2018」铁人两项[圆方树+树形DP]
		
主要卡在一个结论上..关于点双有一个常用结论,也经常作为在圆方树/简单路径上的良好性质,对于任意点双内互不相同的三点$s,c,t$,都存在简单路径$s\to c\to t$,证明不会.可以参见clz博 ...
 - [APIO2018]铁人两项 --- 圆方树
		
[APIO2018] 铁人两项 题目大意: 给定一张图,问有多少三元组(a,b,c)(a,b,c 互不相等)满足存在一条点不重复的以a为起点,经过b,终点为c的路径 如果你不会圆方树 ------- ...
 - [APIO2018] Duathlon 铁人两项 圆方树,DP
		
[APIO2018] Duathlon 铁人两项 LG传送门 圆方树+简单DP. 不会圆方树的话可以看看我的另一篇文章. 考虑暴力怎么写,枚举两个点,答案加上两个点之间的点的个数. 看到题面中的一句话 ...
 - [APIO2018]铁人两项 [圆方树模板]
		
把这个图缩成圆方树,把方点的权值设成-1,圆点的权值设成点双的size,算 经过这个点的路径的数量*这个点的点权 的和即是答案. #include <iostream> #include ...
 - [BZOJ5463][APIO2018]铁人两项(圆方树DP)
		
题意:给出一张图,求满足存在一条从u到v的长度大于3的简单路径的有序点对(u,v)个数. 做了上一题[HDU5739]Fantasia(点双连通分量+DP),这个题就是一个NOIP题了. 一开始考虑了 ...
 - 【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 路 ...
 - 洛谷P4630 铁人两项--圆方树
		
一道很好的圆方树入门题 感谢PinkRabbit巨佬的博客,讲的太好啦 首先是构建圆方树的代码,也比较好想好记 void tarjan(int u) { dfn[u] = low[u] = ++dfn ...
 - 2019.03.29 bzoj5463: [APIO2018] 铁人两项(圆方树+树形dp)
		
传送门 题意简述:给你一张无向图,问你满足存在从a−>b−>ca->b->ca−>b−>c且不经过重复节点的路径的有序点对(a,b,c)(a,b,c)(a,b,c) ...
 
随机推荐
- 效率提升工具Listary
			
效率提升工具Listary https://baijiahao.baidu.com/s?id=1590032175308204846&wfr=spider&for=pc
 - 转载:centos安装gitlab详解
			
原文地址:http://blog.csdn.net/jiangtao_st/article/details/73612298 一, 服务器快速搭建gitlab方法 可以参考gitlab中文社区 的教程 ...
 - 九、分组查询详解(group by & having)
			
本篇内容 分组查询语法 聚合函数 单字段分组 多字段分组 分组前筛选数据 分组后筛选数据 where和having的区别 分组后排序 where & group by & having ...
 - Django中使用geetest验证
			
一.geetest的使用方法 首先需要在setting中配置好文件 GEE_TEST = { "gee_test_access_id": "37ca5631edd1e88 ...
 - CIP 协议安全扫盲
 - RT-Thread--线程间同步
			
线程间同步 一个线程从传感器中接收数据并且将数据写到共享内存中,同时另一个线程周期性的从共享内存中读取数据并发送去显示,下图描述了两个线程间的数据传递: 如果对共享内存的访问不是排他性的,那么各个线程 ...
 - lvs+keepalived集群架构服务
			
一,LVS功能详解 1.1 LVS(Linux Virtual Server)介绍 LVS是Linux Virtual Server 的简写(也叫做IPVS),意即Linux虚拟服务器,是一个虚拟的服 ...
 - Python 'takes exactly 1 argument (2 given)' Python error
			
Python,定义urlConfig 接收参数,正常传递参数时,出现,多给了一个参数的错误问题, 定义class的函数之后,在调用的时候出现“'takes exactly 1 argument (2 ...
 - CentOS 6.5下快速搭建ftp服务器
			
来源:Linux社区 作者:Linux CentOS 6.5下快速搭建ftp服务器 1.用root 进入系统 2.使用命令 rpm -qa|grep vsftpd 查看系统是否安装了ftp,若安装了v ...
 - java疑问
			
1. new String("abc")究竟创建几个对象? 答: 一个或两个, 如果常量池中原来有"abc", 那么只创建一个对象; 如果常量池中原来没有&qu ...