最大半连通子图 bzoj 1093
最大半连通子图
【问题描述】
一个有向图G = (V,E)称为半连通的(Semi-Connected),如果满足:∀ u, v ∈V,满足u—>v 或 v —> u,即对于图中任意两点u,v, 存在一条u到v的有向路径或者从v到u的有向路径。
若满足,则称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.
【样例输入】
6 6 20070603
1 2
2 1
1 3
2 4
5 6
6 4
【样例输出】
3
3
【数据规模】
对于20%的数据, N ≤ 18;
对于60%的数据, N ≤ 10000;
对于100%的数据, N ≤ 100000, M ≤ 1000000;
对于100%的数据, X ≤ 108。
题解:
首先用Tarjon缩点,去重连边,得到新图,那么题目就变成了求图中最长链及最长链个数
最长链可以直接用拓扑排序
最长链个数用一个类似递推的方法
记录每一个点的方案数
那么当前点的方案数就等于连到此点且满足距离相等的点的方案数之和
最后查找距离等于最长链的点,答案为它们的方案数之和
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
using namespace std;
inline int Get()
{
int x = ;
char c = getchar();
while('' > c || c > '') c = getchar();
while('' <= c && c <= '')
{
x = (x << ) + (x << ) + c - '';
c = getchar();
}
return x;
}
const int me = ;
int n, m, mo;
int x[me], y[me];
int tot;
int de[me], to[me], fir[me], nex[me];
int ue[me];
int si[me];
inline void Ins(int x, int y)
{
nex[++tot] = fir[x];
fir[x] = tot;
to[tot] = y;
}
int num, top, col;
int ti[me], lo[me], st[me], co[me];
inline void Tarjan(int u)
{
ti[u] = lo[u] = ++num;
st[++top] = u;
for(int i = fir[u]; i; i = nex[i])
{
int v = to[i];
if(!ti[v])
{
Tarjan(v);
lo[u] = min(lo[u], lo[v]);
}
else
if(!co[v])
lo[u] = min(lo[u], ti[v]);
}
if(lo[u] == ti[u])
{
co[u] = ++col;
++si[col];
while(st[top] != u)
{
++si[col];
co[st[top]] = col;
--top;
}
--top;
}
}
int t, w;
int ans;
int e[me];
int dis[me];
inline bool rule(int a, int b)
{
if(x[a] != x[b]) return x[a] < x[b];
return y[a] < y[b];
}
int nu[me];
inline void Remove()
{
for(int i = ; i <= m; ++i)
{
nu[i] = i;
x[i] = co[x[i]];
y[i] = co[y[i]];
}
sort(nu + , nu + + m, rule);
}
inline void Build()
{
tot = ;
memset(fir, , sizeof(fir));
for(int i = ; i <= m; ++i)
{
int z = nu[i];
if((x[z] != y[z]) && (x[z] != x[nu[i - ]] || y[z] != y[nu[i - ]]))
{
++de[y[z]];
Ins(x[z], y[z]);
}
}
}
inline void Reset()
{
for(int i = ; i <= col; ++i)
if(!de[i])
{
ue[++w] = i;
dis[i] = si[i];
e[i] = ;
if(dis[ans] < dis[i]) ans = i;
}
}
inline void Topo()
{
while(t < w)
{
int u = ue[++t];
for(int i = fir[u]; i; i = nex[i])
{
int v = to[i];
--de[v];
if(dis[v] < dis[u] + si[v])
{
dis[v] = dis[u] + si[v];
e[v] = ;
if(dis[ans] < dis[v]) ans = v;
}
if(dis[v] == dis[u] + si[v])
e[v] = (e[v] + e[u]) % mo;
if(!de[v]) ue[++w] = v;
}
}
}
int anss;
inline void Ask()
{
for(int i = ; i <= n; ++i)
if(dis[i] == dis[ans])
anss = (anss + e[i]) % mo;
}
int main()
{
n = Get(), m = Get(), mo = Get();
for(int i = ; i <= m; ++i)
{
x[i] = Get(), y[i] = Get();
Ins(x[i], y[i]);
}
for(int i = ; i <= n; ++i)
if(!ti[i])
Tarjan(i);
Remove();
Build();
Reset();
Topo();
Ask();
printf("%d\n%d", dis[ans], anss);
}
最大半连通子图 bzoj 1093的更多相关文章
- BZOJ 1093 [ZJOI2007] 最大半连通子图(强联通缩点+DP)
题目大意 题目是图片形式的,就简要说下题意算了 一个有向图 G=(V, E) 称为半连通的(Semi-Connected),如果满足图中任意两点 u v,存在一条从 u 到 v 的路径或者从 v 到 ...
- 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 [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 最大半连通子图 题解
1093: [ZJOI2007]最大半连通子图 Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 2767 Solved: 1095[Submit][S ...
- [BZOJ]1093 最大半连通子图(ZJOI2007)
挺有意思的一道图论. Description 一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:∀u,v∈V,满足u→v或v→u,即对于图中任意两点u,v,存在一条u到v ...
- 【刷题】BZOJ 1093 [ZJOI2007]最大半连通子图
Description 一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:?u,v∈V,满足u→v或v→u,即对于图中任意 两点u,v,存在一条u到v的有向路径或者从v到 ...
- bzoj1093: [ZJOI2007]最大半连通子图 scc缩点+dag上dp
一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:?u,v∈V,满足u→v或v→u,即对于图中任意两点u,v,存在一条u到v的有向路径或者从v到u的有向路径.若G'=(V ...
随机推荐
- 著名ERP厂商的SSO单点登录解决方案介绍一
SSO英文全称Single Sign On,单点登录.SSO是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统.它包括可以将这次主要的登录映射到其他应用中用于同一个用户 ...
- Impress.js上手 - 抛开PPT、制作Web 3D幻灯片放映
前言: 如果你已经厌倦了使用PPT设置路径.设置时间.设置动画方式来制作动画特效.那么Impress.js将是你一个非常好的选择. 用它制作的PPT将更加直观.效果也是嗷嗷美观的. 当然,如果用它来装 ...
- 树莓派3B的食用方法-1(装系统 网线ssh连接)
首先要有一个树莓派3B , 在某宝买就行, 这东西基本上找到假货都难,另外国产和英国也没什么差别,差不多哪个便宜买哪个就行. 不要买店家的套餐,一个是配的东西有些不需要,有的质量也不好. 提示:除了G ...
- python_单元测试unittest
Python自带一个单元测试框架是unittest模块,用它来做单元测试,它里面封装好了一些校验返回的结果方法和一些用例执行前的初始化操作. 步骤1:首先引入unittest模块--import un ...
- SVG:textPath深入理解
SVG的文本可以沿着一条自定义的Path来排布,比如曲线.圆形等等,使用方式如下所示(来源MDN): <svg viewBox="0 0 1000 300" xmlns=&q ...
- ABP源码分析三:ABP Module
Abp是一种基于模块化设计的思想构建的.开发人员可以将自定义的功能以模块(module)的形式集成到ABP中.具体的功能都可以设计成一个单独的Module.Abp底层框架提供便捷的方法集成每个Modu ...
- TODO:Go语言同名Go字体发布
TODO:Go语言同名Go字体发布 2016-11-16 Go语言官方博客发布了一款同名字体–Go字体.此字体族包括正常.粗体和斜体渲染,支持比例和等宽字体.此字体已经经过用于编程方面的技术测试,使用 ...
- Entity Framework 6 Recipes 2nd Edition(11-5)译 -> 从”模型定义”函数返回一个匿名类型
11-5. 从”模型定义”函数返回一个匿名类型 问题 想创建一个返回一个匿名类型的”模型定义”函数 解决方案 假设已有游客(Visitor) 预订(reservation)房间(hotel ) 的模型 ...
- 火狐浏览器中event不起作用解决办法--记录(一)
今天遇到了这个问题.IE,谷歌下都没问题,但在FF下却不起作用,很郁闷查了半天,看别人博文写了老长,结果试了要么起作用,但太麻烦,要么不起作用,说了那么多跟没说一样. 其实只要这一句代码就行:e=ar ...
- Laravel学习--关于Relation的坑
前段时间比较忙,就没有坚持写博客,但发现这周末再想捡起来,好难,一直到了今天晚上,才决定坐下来写一篇,哈哈哈-- 最近在用 Laravel 5.2,踩了几个关于 Relation 的坑,在这里用博客记 ...