题解 P2272 【[ZJOI2007]最大半连通子图】
P2272 [ZJOI2007]最大半连通子图
萌新初学Tarjan,在《信息学奥赛一本通-提高篇》中看到这题,看到题解不多,便想发布一篇较为清新简洁的题解。——第5道紫题
题目大意:
定义最大半连通图:对于图中任意两点u,v,存在一条u到v的有向路径 或者 从v到u的有向路径。求一个图中不同的最大半连通子图的数目。
看到题面时大家很容易想到,如果两点互相可以到达,那么它们必是半连通图,所以考虑先Tarjan缩点(P3387 【模板】缩点(Tarjan缩点+DAGdp))
接着去除重边重新建图,你会发现,在这个有向无环图(DAG)中,半连通子图都是一条链(可以举反例试试,这条链不可能有分支,否则将有两点无法抵达另一方)
于是,G的最大半连通子图拥有的节点数K就是最长链长度,不同的最大半连通子图的数目就是最长链个数。
信息学一本通:最长链可以直接用拓扑排序(topo),最长链个数用一个类似DP的方法,用f【i】表示以 i 为终点的方案数,那么f【i】就等于满足距离为起点到 i 的临时最短距离的点的 f 的和。然后查找距离等于最长链的点,答案为它们的方案数之和
其他题解中已经给出了拓扑的算法,我借鉴大佬的程序用的是搜索,先一直搜到终点再回来更新答案。由于数据范围#7一直RE,后来改为const int N=1e5+5,M=2e6+5;终于AC。。qwq高性能。。
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=1e5+5,M=2e6+5;
bool f[N];
//f 在搜索中判断是否走过
int n,m,mod,now,d[N],a[N],ans,maxans,ch[N];
//d指从u到终点的最长链距离,a指最长链点数,ch指出度
int h[N],u[M],v[M],r[N],nu[M],cnt;
//h是链式前向星的建边head,u,v保存初始读入的边左右两点,nu存初始时边的编号,r是入度
int top,co,dfn[N],low[N],c[N],s[N],st[N];
//dfn,low,st用于Tarjan,c表所在强连通分量编号,s指所在强连通分量点数
struct edge {
int h,to;
} e[M];
#define rint register int
#define min(a,b) (a<b? a:b)
#define max(a,b) (a>b? a:b)
inline bool cmp(int a,int b) {
return u[a]<u[b] || (u[a]==u[b] && v[a]<v[b]);
}//将边排序,方便重新建图
inline void add(int u,int v) {
e[++cnt].h=h[u],h[u]=cnt,e[cnt].to=v;
}
inline int read() {
int w=1,ans=0;
char ch=getchar();
while(ch>'9'||ch<'0') if (ch=='-') w=-1,ch=getchar();
while(ch<='9'&& ch>='0') ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans*w;
}
inline void Tarjan(int u) {
dfn[u]=low[u]=++now;
st[++top]=u;
for (rint i=h[u]; i; i=e[i].h) {
int v=e[i].to;
if (!dfn[v])
Tarjan(v),low[u]=min(low[u],low[v]);
else if (!c[v])
low[u]=min(low[u],dfn[v]);
}
if (low[u]==dfn[u]) {
c[u]=++co,s[co]++;
while(st[top]!=u)
s[co]++,c[st[top]]=co,top--;
top--;
}
}//标准缩点
inline void dfs(int u) {
f[u]=1;
if (!ch[u]) {//如果没有出度,即到头了
d[u]=s[u],a[u]=1;//距离为点数,以u为起点方案为1
maxans=max(maxans,d[u]);//更新最长链距离
return;
}
for (rint i=h[u]; i; i=e[i].h) {
int v=e[i].to;
if (!f[v]) dfs(v);//继续搜索链的后面
if (d[v]+s[u]>d[u])//若以u为起点的链距离可以更长
d[u]=d[v]+s[u],a[u]=a[v]%mod;//更新
else if (d[u]==d[v]+s[u])//若最长链距离相同
a[u]=(a[u]+a[v])%mod;//加上方案数
maxans=max(maxans,d[u]);
}
}
int main() {
n=read(),m=read(),mod=read();
for (rint i=1; i<=m; i++) u[i]=read(),v[i]=read(),add(u[i],v[i]);
for (rint i=1; i<=n; i++) if (!dfn[i]) Tarjan(i);
cnt=0;
memset(h,0,sizeof h);
memset(e,0,sizeof e);
for (rint i=1; i<=m; i++)
nu[i]=i,u[i]=c[u[i]],v[i]=c[v[i]];
sort(nu+1,nu+m+1,cmp);//按u,v排序边
for (rint i=1; i<=m; i++)
{
int num=nu[i];
if (u[num]!=v[num] && (u[num]!=u[nu[i-1]] || v[num]!=v[nu[i-1]]))//若此边不是自环,且与上一条边不同(去除重边)
++ch[u[num]],++r[v[num]],add(u[num],v[num]);}
//出度入度加1,加边
for (rint i=1; i<=co; i++) if (!r[i] && !f[i]) dfs(i);//入度为0且未搜索过
for (rint i=1; i<=co; i++) if (d[i]==maxans) ans=(ans+a[i])%mod;//统计答案
printf("%d\n%d\n",maxans,ans);
}
题解 P2272 【[ZJOI2007]最大半连通子图】的更多相关文章
- 洛谷 P2272 [ZJOI2007]最大半连通子图 解题报告
P2272 [ZJOI2007]最大半连通子图 题目描述 一个有向图\(G=(V,E)\)称为半连通的\((Semi-Connected)\),如果满足:\(\forall u,v \in V\),满 ...
- Luogu P2272 [ZJOI2007]最大半连通子图(Tarjan+dp)
P2272 [ZJOI2007]最大半连通子图 题意 题目描述 一个有向图\(G=(V,E)\)称为半连通的\((Semi-Connected)\),如果满足:\(\forall u,v\in V\) ...
- luogu P2272 [ZJOI2007]最大半连通子图
题目描述 一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:?u,v∈V,满足u→v或v→u,即对于图中任意两点u,v,存在一条u到v的有向路径或者从v到u的有向路径.若 ...
- P2272 [ZJOI2007]最大半连通子图 tarjan+DP
思路:$tarjan+DP$ 提交:1次 题解:首先对于一个强连通分量一定是一个半连通分量,并且形成的半连通分量的大小一定是它的$size$,所以我们先缩点. 这样,我们相当于要在新的$DAG$上找一 ...
- P2272 [ZJOI2007]最大半连通子图
思路 tarjan的题目 注意是要选出一个点集而不是边集 第一问就是缩点之后最长链,第二问就是有多少个最长链,注意缩点后连边要去重(不然一个链的方案可能会被统计多次) 代码 #include < ...
- BZOJ 1093 [ZJOI2007]最大半连通子图
1093: [ZJOI2007]最大半连通子图 Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 1986 Solved: 802[Submit][St ...
- BZOJ 1093 [ZJOI2007] 最大半连通子图(强联通缩点+DP)
题目大意 题目是图片形式的,就简要说下题意算了 一个有向图 G=(V, E) 称为半连通的(Semi-Connected),如果满足图中任意两点 u v,存在一条从 u 到 v 的路径或者从 v 到 ...
- 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)的结点数. 其实就 ...
随机推荐
- 通过异步程序调用(APC)实现的定时功能
定时器是一个在特定时间或者规则间隔被激发的内核对象.结合定时器的异步程序调用可以允许回调函数在任何定时器被激发的时候执行.本文的例子代码显示了如何实现. 使用本定时器时,你需要把常量_WIN32_WI ...
- LigerUI中Grid的使用时关于url请求不到数据的问题
前台代码:(这里贴的是js的代码,完整的代码可以在LigerUI的文档中找到), 这里使用的是url请求数据,问题不是处在前台,所以就不细说. $("#maingrid").lig ...
- Tobject 类解析
TObject = class //创建 constructor Create; //释放 procedure Free; //初始化实列 class functi ...
- 快速搭建多线程Windows服务解决方案
一.引言 在软件开发过程中windows服务有的时候非常有用,用于同步数据,发送邮件,宿主WF引擎服务等,但是快速搭建一个好用多线程进行多任务处理的程序往往是一个项目必须考虑的问题.自己在项目中也经常 ...
- 使用dumpbin命令查看dll导出函数及重定向输出到文件(VS自带)
以前查看dll导出函数,一般使用Viewdll等第三方工具.但由于Viewdll采用dephi编写,因此仅能查看32位的dll.其实微软已经帮我们提供一个查看dll导出函数的命令,嵌在VS开发环境中, ...
- Linux下C/C++帮助手册安装方法
1. 安装C的帮助手册 如果你使用的Linux发行版, 默认没有安装C语言的库函数MAN手册, 使用下面的方法解决: # sudo apt-get install manpages # sudo a ...
- 自己总结OpenSSL的变化
经过查看openssl源码自带的Makefile,发现: 1) 从0.9.7开始 https://www.openssl.org/source/old/0.9.x/openssl-0.9.7k.tar ...
- Zabbix监控ActiveMQ
当我们在线上使用了ActiveMQ 后,我们需要对一些参数进行监控,比如 消息是否有阻塞,哪个消息队列阻塞了,总的消息数是多少等等.下面我们就通过 Zabbix 结合 Python 脚本来实现对 Ac ...
- Spring之bean生命始末
可以为Bean定制初始化后的生命行为,也可以为Bean定制销毁前的生命行为.举例:ba06包.首先,这些方法需要在Bean类中事先定义好:是方法名随意的public void方法. 其次,在配置文件的 ...
- C语言指针学习总结
上学的时候学习C语言,最烦的就是里面指针,可是指针也恰恰是C语言的灵魂. 最近在重温数据结构的内容,因为大多数据结构的教材都是用C语言描述的,而数据结构中也大量的用到了指针的内容,所以我就在这篇笔记中 ...