[BZOJ]1093 最大半连通子图(ZJOI2007)
挺有意思的一道图论。
Description
一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:∀u,v∈V,满足u→v或v→u,即对于图中任意两点u,v,存在一条u到v的有向路径或者从v到u的有向路径。若G'=(V',E')满足V'⊆V,E'是E中所有跟V'有关的边,则称G'是G的一个导出子图。若G'是G的导出子图,且G'半连通,则称G'为G的半连通子图。若G'是G所有半连通子图
中包含节点数最多的,则称G'是G的最大半连通子图。给定一个有向图G,请求出G的最大半连通子图拥有的节点数K,以及不同的最大半连通子图的数目C。由于C可能比较大,仅要求输出C对X的余数。
Input
第一行包含两个整数N,M,X。N,M分别表示图G的点数与边数,X的意义如上文所述接下来M行,每行两个正整数a, b,表示一条有向边(a, b)。图中的每个点将编号为1,2,3…N,保证输入中同一个(a,b)不会出现两次。
Output
应包含两行,第一行包含一个整数K。第二行包含整数C Mod X。
Sample Input
6 6 20070603
1 2
2 1
1 3
2 4
5 6
6 4
Sample Output
3
3
HINT
N ≤10000, M ≤1000000,X ≤10^8。
Solution
拿到这题,我们首先思考半连通分量是个什么东西。
首先我们知道,强连通分量一定是半连通分量,
题目要求我们求最大的半连通分量,所以如果选取了一个强连通分量里的点,那么把该点所在的整个强连通分量都选进去肯定没问题,选取一个强连通分量和选取一个点是等价的。
所以我们很自然地用tarjan缩了缩点……
然后我们得到了一个带点权的拓扑图。
仔细一想,我们发现拓扑图中的半连通分量是一条链,
所以问题也就变成了找拓扑图中的最长链,并统计最长链的条数。
(这个用DP不会做你退群吧)
注意缩点之后要处理掉重边。时间复杂度O(n+m)。
#include <cstdio>
#include <algorithm>
#include <cstring>
#define MN 100005
#define MM 1000005
using namespace std;
struct edge{int nex,to;}e[MM];
struct bian{int x,y;}b[MM];
bool u[MN],ink[MN];
int low[MN],st[MN],hr[MN],bel[MN],w[MN],d[MN],q[MN],f1[MN],f2[MN];
int dfn,pin,tp,n,m,mod,hd,tl,ans1,ans2;
char B[<<],*SS,*TT; inline char getc() {return SS==TT&&(TT=(SS=B)+fread(B,,<<,stdin),SS==TT)?EOF:*SS++;}
inline int read()
{
register int n=; char c;
do c=getc(); while (c<'' || c>'');
do n=n*+c-'',c=getc(); while (c>='' && c<='');
return n;
} inline void ins(int x,int y) {e[++pin]=(edge){hr[x],y}; hr[x]=pin;} void tarjan(int x)
{
register int i,lt;
low[x]=lt=++dfn;
u[x]=ink[x]=true; st[++tp]=x;
for (i=hr[x];i;i=e[i].nex)
{
if (u[e[i].to]&&!ink[e[i].to]) continue;
if (!u[e[i].to]) tarjan(e[i].to);
low[x]=min(low[x],low[e[i].to]);
}
if (low[x]==lt)
for (;st[tp+]!=x;--tp) ink[st[tp]]=false,bel[st[tp]]=x,++w[x];
} int main()
{
register int i,x;
n=read(); m=read(); mod=read();
for (i=;i<=m;++i) b[i].x=read(),b[i].y=read(),ins(b[i].x,b[i].y);
for (i=;i<=n;++i) if (!u[i]) tarjan(i);
memset(hr,,sizeof(hr)); pin=;
for (i=;i<=m;++i)
if (bel[b[i].x]!=bel[b[i].y])
ins(bel[b[i].x],bel[b[i].y]),++d[bel[b[i].y]];
for (i=hd=;i<=n;++i) if (bel[i]==i&&!d[i]) q[++tl]=i,f1[i]=w[i],f2[i]=;
for (;hd<=tl;++hd)
{
for (x=q[hd],i=hr[x];i;i=e[i].nex)
{
if (!--d[e[i].to]) q[++tl]=e[i].to;
if (!u[e[i].to]) continue; else u[e[i].to]=false;
if (f1[x]+w[e[i].to]>f1[e[i].to]) f1[e[i].to]=f1[x]+w[e[i].to],f2[e[i].to]=f2[x];
else if (f1[x]+w[e[i].to]==f1[e[i].to]) f2[e[i].to]+=f2[x],f2[e[i].to]-=f2[e[i].to]>=mod?mod:;
}
for (i=hr[x];i;i=e[i].nex) u[e[i].to]=true;
}
for (i=;i<=n;++i)
if (f1[i]>ans1) ans1=f1[i],ans2=f2[i];
else if (f1[i]==ans1) ans2+=f2[i],ans2-=ans2>=mod?mod:;
printf("%d\n%d",ans1,ans2);
}
Last Word
有向图用tarjan缩完点得到的是拓扑图,无向图缩环会变成树(森林)。
用了n+e光速读入后立竿见影地卡到了这道题的rank2。(代码画风崩坏不可避)
[BZOJ]1093 最大半连通子图(ZJOI2007)的更多相关文章
- bzoj 1093 最大半连通子图 - Tarjan - 拓扑排序 - 动态规划
一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:?u,v∈V,满足u→v或v→u,即对于图中任意两点u,v,存在一条u到v的有向路径或者从v到u的有向路径.若G'=(V ...
- BZOJ 1093 最大半连通子图 题解
1093: [ZJOI2007]最大半连通子图 Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 2767 Solved: 1095[Submit][S ...
- BZOJ 1093 最大半连通子图
缩点求最长链. #include<iostream> #include<cstdio> #include<cstring> #include<algorith ...
- Tarjan水题系列(5):最大半连通子图 [ZJOI2007 luogu P2272]
题目 大意: 缩点后转为求最长链的长度和最长链的个数 思路: 看懂题就会做系列 长度和个数都可以拓扑排序后DP求得 毕竟是2007年的题 代码: 如下 #include <cstdio> ...
- BZOJ 1093 [ZJOI2007] 最大半连通子图(强联通缩点+DP)
题目大意 题目是图片形式的,就简要说下题意算了 一个有向图 G=(V, E) 称为半连通的(Semi-Connected),如果满足图中任意两点 u v,存在一条从 u 到 v 的路径或者从 v 到 ...
- BZOJ 1093 [ZJOI2007]最大半连通子图
1093: [ZJOI2007]最大半连通子图 Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 1986 Solved: 802[Submit][St ...
- bzoj 1093 [ZJOI2007]最大半连通子图(scc+DP)
1093: [ZJOI2007]最大半连通子图 Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 2286 Solved: 897[Submit][St ...
- BZOJ 1093: [ZJOI2007]最大半连通子图( tarjan + dp )
WA了好多次... 先tarjan缩点, 然后题意就是求DAG上的一条最长链. dp(u) = max{dp(v)} + totu, edge(u,v)存在. totu是scc(u)的结点数. 其实就 ...
- 【刷题】BZOJ 1093 [ZJOI2007]最大半连通子图
Description 一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:?u,v∈V,满足u→v或v→u,即对于图中任意 两点u,v,存在一条u到v的有向路径或者从v到 ...
随机推荐
- 【iOS】OC-Quartz2D简单使用
什么是Quartz2D Quartz 2D是一个二维绘图引擎,同时支持iOS和Mac系统 作用 ? 1 2 3 4 5 6 7 8 9 <code>Quartz 2D能完成的工作 绘制图形 ...
- 消除ExtJS6的extjs-trila字样
- windows安装虚拟主机virtualbox遇到的困难
本来想到可以在windows安装虚拟主机virtualbox,但是怎么自己的windows是盗版的,由于主题已经被破解了,所以不能安装结果强制性的进入pe然后从网上下载的dll文件复制到 c/wind ...
- Docker学习笔记 - Docker容器的日志
docker logs [-f] [-t] [--tail] 容器名 -f -t --tail="all" 无参数:返回所有日志 -f 一直跟踪变化并返回 -t 带时间戳返 ...
- python 开发之路 -MySQL
阅读目录 第一篇 : 数据库 之 基本概念 第二篇 : MySQL 之 库操作 第三篇 : MySQL 之 表操作 第四篇 : MySQL 之 数据操作 第五篇 : MySQL 之 视图.触发器.存储 ...
- POJ-1789 Truck History---最小生成树Prim算法
题目链接: https://vjudge.net/problem/POJ-1789 题目大意: 用一个7位的string代表一个编号,两个编号之间的distance代表这两个编号之间不同字母的个数.一 ...
- hdu1051 Wooden Sticks---贪心
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1051 题目大意:给你n根木棍的长度和重量.根据要求求出制作该木棍的最短时间.建立第一个木棍需要1分钟 ...
- 1.4WEB API 路由配置及参数传输
在没有添加路由时,webapi 的访问比较恶心,访问的时候是没有接口方法的.如果定义了同样参数的方法,它就傻傻的分不清的,为了解决这个问题,我们加上路由吧. 这是生成的api帮助文档列表,可以看到下面 ...
- Text-鼠标点击事件
from tkinter import * import webbrowser master=Tk() text=Text(master,width=50,height=20) text.pack() ...
- this对象指向
this表示函数运行时,自动生成的一个内部对象,只能在函数内部运行 function test(){ this.x = 1; } 随着使用场景的变化,this的值会发生变化 原则:this指的值调用函 ...