洛谷 P2272 [ZJOI2007]最大半连通子图 解题报告
P2272 [ZJOI2007]最大半连通子图
题目描述
一个有向图\(G=(V,E)\)称为半连通的\((Semi-Connected)\),如果满足:\(\forall u,v \in V\),满足\(u \to v\)或\(v \to u\),即对于图中任意两点\(u\),\(v,\)存在一条\(u\)到\(v\)的有向路径或者从\(v\)到\(u\)的有向路径。若\(G'=(V',E')\)满足\(V' \in V\),\(E'\)是\(E\)中所有跟\(V'\)有关的边,则称\(G'\)是\(G\)的一个导出子图。若\(G'\)是\(G\)的导出子图,且\(G'\)半连通,则称\(G'\)为\(G\)的半连通子图。若\(G'\)是\(G\)所有半连通子图中包含节点数最多的,则称\(G'\)是\(G\)的最大半连通子图。给定一个有向图\(G\),请求出\(G\)的最大半连通子图拥有的节点数\(K\),以及不同的最大半连通子图的数目\(C\)。由于\(C\)可能比较大,仅要求输出\(C\)对\(X\)的余数。
输入输出格式
输入格式:
第一行包含两个整数\(N\),\(M\),\(X\)。\(N\),\(M\)分别表示图\(G\)的点数与边数,X的意义如上文所述接下来\(M\)行,每行两个正整数\(a\),\(b\),表示一条有向边\((a,b)\)。图中的每个点将编号为\(1,2,3…N\),保证输入中同一个\((a,b)\)不会出现两次。
输出格式:
应包含两行,第一行包含一个整数\(K\)。第二行包含整数\(C\) \(Mod\) \(X\).
说明
对于100%的数据, \(N \le 100000, M \le 1000000, X \le 10^8\)
这种题先缩点都成套路了吧
考虑在一个有向无环图中半联通图是什么,结果是一条链,直接做DP就行了
但是!!
这个题一定要考虑重边
Code:
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#define ll long long
using namespace std;
const int N=100010;
const int M=1000010;
int head0[N],Next0[M],to0[M],cnt0;
void add0(int u,int v)
{
Next0[++cnt0]=head0[u];to0[cnt0]=v;head0[u]=cnt0;
}
int head[N],Next[M],to[M],cnt;
void add(int u,int v)
{
Next[++cnt]=head[u];to[cnt]=v;head[u]=cnt;
}
int dfn[N],low[N],in[N],s[N],ha[N],siz[N],time,tot;
int n,m,n_,cntt;
ll p;
pair <int,int > e[M];
void tarjan(int now)
{
dfn[now]=low[now]=++time;
s[++tot]=now;
in[now]=1;
for(int i=head0[now];i;i=Next0[i])
{
int v=to0[i];
if(!dfn[v])
{
tarjan(v);
low[now]=min(low[now],low[v]);
}
else if(in[v])
low[now]=min(low[now],dfn[v]);
}
if(low[now]==dfn[now])
{
int k,Siz=0;
n_++;
do
{
k=s[tot--];
Siz++;
ha[k]=n_;
in[k]=0;
}while(k!=now);
siz[n_]=Siz;
}
}
void New()
{
for(int i=1;i<=n;i++)
{
for(int j=head0[i];j;j=Next0[j])
{
int v=to0[j];
if(ha[v]!=ha[i])
e[++cntt]=make_pair(ha[i],ha[v]);
}
}
sort(e+1,e+1+cntt);
cntt=unique(e+1,e+1+cntt)-(e+1);
for(int i=1;i<=cntt;i++)
{
in[e[i].second]++;
add(e[i].first,e[i].second);
}
}
void init()
{
scanf("%d%d%lld",&n,&m,&p);
int u,v;
for(int i=1;i<=m;i++)
{
scanf("%d%d",&u,&v);
add0(u,v);
}
for(int i=1;i<=n;i++)
if(!dfn[i])
tarjan(i);
memset(in,0,sizeof(in));
New();
}
ll Cnt[N],ans;
int dp[N],mx;
queue <int > q;
void work()
{
for(int i=1;i<=n_;i++)
if(!in[i])
{
q.push(i);
dp[i]=siz[i];
mx=max(mx,dp[i]);
Cnt[i]=1;
}
while(!q.empty())
{
int u=q.front();
q.pop();
for(int i=head[u];i;i=Next[i])
{
int v=to[i];
if(dp[v]<dp[u]+siz[v])
{
dp[v]=dp[u]+siz[v];
Cnt[v]=Cnt[u];
mx=max(mx,dp[v]);
}
else if(dp[v]==dp[u]+siz[v])
(Cnt[v]+=Cnt[u])%=p;
in[v]--;
if(!in[v]) q.push(v);
}
}
for(int i=1;i<=n_;i++)
if(dp[i]==mx)
(ans+=Cnt[i])%=p;
printf("%d\n%lld\n",mx,ans);
}
int main()
{
init();
work();
return 0;
}
2018.7.26
洛谷 P2272 [ZJOI2007]最大半连通子图 解题报告的更多相关文章
- BZOJ1093或洛谷2272 [ZJOI2007]最大半连通子图
BZOJ原题链接 洛谷原题链接 和 Going from u to v or from v to u?(题解)这道题类似,只不过是求最大子图的大小和个数而已. 一样用\(tarjan\)求强连通分量, ...
- 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 < ...
- 题解 P2272 【[ZJOI2007]最大半连通子图】
P2272 [ZJOI2007]最大半连通子图 萌新初学Tarjan,在<信息学奥赛一本通-提高篇>中看到这题,看到题解不多,便想发布一篇较为清新简洁的题解.--第5道紫题 题目大意: 定 ...
- 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 ...
随机推荐
- ACID、数据库隔离级别
ACID: A(Atomicity):原子性,要么全部执行,要么都不执行 C(consistency):一致性: 特点: 1.一个操作除法级联,这些必须成功,否则全部失败(原子性) 2.所有节点同步更 ...
- 监控系统cpu相关统计信息
背景:需要测试监控各个操作系统平台机器上的cpu相关的各种统计信息 为了方便测试,我写了一个比较通用的shell脚本,目前可以兼容Redhat6+,Redhat7+,其他操作系统没测,可以实时监控机器 ...
- 第五模块:WEB开发基础 第2章·JavaScript基础
01-JavaScript的历史发展过程 02-js的引入方式和输出 03-命名规范和变量的声明定义 04-五种基本数据类型 05-运算符 06-字符串处理 07-数据类型转换 08-流程控制语句if ...
- Siki_Unity_1-3_Unity零基础入门_古迹探险
1-3 Unity零基础入门 古迹探险 任务1/2:资料下载 链接:https://pan.baidu.com/s/1jHVymNk 密码:rbob 任务3:工程的创建和打开 Project:古迹探险 ...
- 微信小程序之基础案例详细解释
这是案例的初始页面 首先关于这个案例上面的app.json做一个特别详细的解释 首先提醒一下.json文件不能有注释,因此如果要复制的话,请把注释删去 //关于app.json详细学习 { //pag ...
- fastCMS八大核心对象
fastCMS内置system对象,该对象包含八大核心对象,应用于不同的操作场景,分别是: 1.system.string 对象(处理字符类操作) 2.system.number 对象(处理数字类操作 ...
- leetcode-三数之和(java)
三数之和 给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组. 注意:答案中不可 ...
- 一个简单的页面弹窗插件 jquery.pageMsgFrame.js
页面弹窗是网站中常用的交互效果,它可以强提示网站的某些信息给用户,或者作用于某些信息的修改等等功能. 这几天在做一个项目的时候,就顺捎把这个插件写一下,栽棵树,自己乘凉吧. 原创博文,转载请注明出处: ...
- 火狐metamask账号
火狐metamask lock trophy pyramid sunny aim inmate body sense sing castle cinnamon cram
- 应用Response.Write实现带有进度条的多文件上传
前几天,写过一篇随笔“使用RESPONSE.WRITE实现在页面的生命周期中前后台的交互”.说是交互,实际上也主要是在ASP.NET的页面周期中 从后台利用RESPONSE.WRITE向前台即时的推送 ...