【bzoj1093】[ZJOI2007]最大半连通子图 Tarjan+拓扑排序+dp

题目描述

一个有向图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的余数。

输入

第一行包含两个整数N,M,X。N,M分别表示图G的点数与边数,X的意义如上文所述接下来M行,每行两个正整数a, b,表示一条有向边(a, b)。图中的每个点将编号为1,2,3…N,保证输入中同一个(a,b)不会出现两次。N ≤100000, M ≤1000000;对于100%的数据, X ≤10^8

输出

应包含两行,第一行包含一个整数K。第二行包含整数C Mod X.

样例输入

6 6 20070603
1 2
2 1
1 3
2 4
5 6
6 4

样例输出

3
3


题解

Tarjan+拓扑排序+dp

显然,如果原图是一个DAG,那么选择的就是一条链,答案就是最长链;

如果不是呢?Tarjan缩点,然后拓扑排序+dp求带权最长链即可。正确性显然。

需要注意的是缩完点后如果有重边需要只考虑一条的贡献,因为确定了点就确定了边的选择,只有一次转移的机会。

时间复杂度 $O(n+m)$ 

#include <queue>
#include <cstdio>
#include <cctype>
#include <vector>
#include <cstring>
#define N 100010
using namespace std;
queue<int> q;
vector<int> e[N] , v[N];
int p , deep[N] , low[N] , tot , ins[N] , sta[N] , top , bl[N] , si[N] , num , ind[N] , last[N];
struct data
{
	int x , y;
	data(int a = 0 , int b = 0) {x = a , y = b;}
	data operator+(int a) {return data(x + a , y);}
	data operator^(data a)
	{
		if(x > a.x) return *this;
		else if(x < a.x) return a;
		else return data(x , (y + a.y) % p);
	}
}f[N];
void tarjan(int x)
{
	vector<int>::iterator i;
	deep[x] = low[x] = ++tot , ins[x] = 1 , sta[++top] = x;
	for(i = e[x].begin() ; i != e[x].end() ; i ++ )
	{
		if(!deep[*i]) tarjan(*i) , low[x] = min(low[x] , low[*i]);
		else if(ins[*i]) low[x] = min(low[x] , deep[*i]);
	}
	if(deep[x] == low[x])
	{
		int t;
		num ++ ;
		do
		{
			t = sta[top -- ] , ins[t] = 0;
			bl[t] = num , si[num] ++ ;
		}while(t != x);
	}
}
void solve(int n)
{
	vector<int>::iterator i;
	data ans;
	int x;
	for(x = 1 ; x <= n ; x ++ )
		for(i = e[x].begin() ; i != e[x].end() ; i ++ )
			if(bl[x] != bl[*i])
				v[bl[x]].push_back(bl[*i]) , ind[bl[*i]] ++ ;
	for(x = 1 ; x <= num ; x ++ )
		if(!ind[x])
			f[x] = data(si[x] , 1) , q.push(x);
	while(!q.empty())
	{
		x = q.front() , q.pop() , ans = ans ^ f[x];
		for(i = v[x].begin() ; i != v[x].end() ; i ++ )
		{
			if(last[*i] != x) last[*i] = x , f[*i] = f[*i] ^ (f[x] + si[*i]);
			ind[*i] -- ;
			if(!ind[*i]) q.push(*i);
		}
	}
	printf("%d\n%d\n" , ans.x , ans.y);
}
inline char nc()
{
	static char buf[100000] , *p1 , *p2;
	return p1 == p2 && (p2 = (p1 = buf) + fread(buf , 1 , 100000 , stdin) , p1 == p2) ? EOF : *p1 ++ ;
}
inline int read()
{
	int ret = 0; char ch = nc();
	while(!isdigit(ch)) ch = nc();
	while(isdigit(ch)) ret = ((ret + (ret << 2)) << 1) + (ch ^ '0') , ch = nc();
	return ret;
}
int main()
{
	int n = read() , m = read() , i , x , y;
	p = read();
	for(i = 1 ; i <= m ; i ++ ) x = read() , y = read() , e[x].push_back(y);
	for(i = 1 ; i <= n ; i ++ )
		if(!deep[i])
			tarjan(i);
	solve(n);
	return 0;
}

 

 

posted @ 2017-11-24 09:06  GXZlegend  阅读(441)  评论(0编辑  收藏  举报