[CTSC2008]祭祀
题目描述
在遥远的东方,有一个神秘的民族,自称Y族。他们世代居住在水面上,奉龙王为神。每逢重大庆典, Y族都会在水面上举办盛大的祭祀活动。我们可以把Y族居住地水系看成一个由岔口和河道组成的网络。每条河道连接着两个岔口,并且水在河道内按照一个固定的方向流动。显然,水系中不会有环流(下图描述一个环流的例子)。
由于人数众多的原因,Y族的祭祀活动会在多个岔口上同时举行。出于对龙王的尊重,这些祭祀地点的选择必须非常慎重。准确地说,Y族人认为,如果水流可以从一个祭祀点流到另外一个祭祀点,那么祭祀就会失去它神圣的意义。族长希望在保持祭祀神圣性的基础上,选择尽可能多的祭祀的地点。
输入输出格式
输入格式:
第一行包含两个用空格隔开的整数N、M,分别表示岔口和河道的数目,岔口从1到N编号。
接下来M行,每行包含两个用空格隔开的整数u、v,描述一条连接岔口u和岔口v的河道,水流方向为自u向v。
输出格式:
第一行包含一个整数K,表示最多能选取的祭祀点的个数。 接下来一行输出一种可行的选取方案。对于每个岔口依次输出一个整数,如果在该岔口设置了祭祀点,那么输出一个1,否则输出一个0。应确保你输出的1 的个数最多,且中间没有空格。 接下来一行输出,在选择最多祭祀点的前提下,每个岔口是否能够设置祭祀点。对于每个岔口依次输出一个整数,如果在该岔口能够设置祭祀点,那么输出一个 1,否则输出一个0。 注意:多余的空格和换行可能会导致你的答案被判断为错误答案。
输入输出样例
说明
N ≤ 100 M ≤ 1 000
在样例给出的水系中,不存在一种方法能够选择三个或者三个以上的祭祀点。包含两个祭祀点的测试点的方案有两种:
选择岔口1与岔口3(如样例输出第二行),选择岔口1与岔口4。
水流可以从任意岔口流至岔口2。如果在岔口2建立祭祀点,那么任意其他岔口都不能建立祭祀点但是在最优的一种祭祀点的选取方案中我们可以建立两个祭祀点,所以岔口2不能建立祭祀点。对于其他岔口至少存在一个最优方案选择该岔口为祭祀点,所以输出为1011。
没有spj,不用输出后面那些的,只要输出最大值就可以了
这题涉及
最长反链与最小链覆盖
1 #include<iostream> 2 #include<cstdio> 3 #include<algorithm> 4 #include<cmath> 5 #include<cstring> 6 #include<algorithm> 7 using namespace std; 8 struct Node 9 { 10 int next,to; 11 }edge[200001]; 12 int n,m,head[501],match[501],vis[501],num,map[201][201],ans; 13 void add(int u,int v) 14 { 15 num++; 16 edge[num].next=head[u]; 17 head[u]=num; 18 edge[num].to=v; 19 } 20 bool dfs(int x) 21 {int i; 22 for (i=head[x];i;i=edge[i].next) 23 { 24 int v=edge[i].to; 25 if (vis[v]) continue; 26 vis[v]=1; 27 if (match[v]==-1||dfs(match[v])) 28 { 29 match[v]=x; 30 return 1; 31 } 32 } 33 return 0; 34 } 35 int main() 36 {int i,j,u,v,k; 37 cin>>n>>m; 38 for (i=1;i<=m;i++) 39 { 40 scanf("%d%d",&u,&v); 41 map[u][v]=1; 42 } 43 for (k=1;k<=n;k++) 44 for (i=1;i<=n;i++) 45 for (j=1;j<=n;j++) 46 map[i][j]|=map[i][k]&map[k][j]; 47 for (i=1;i<=n;i++) 48 for (j=1;j<=n;j++) 49 if (map[i][j]) add(i,n+j),add(n+j,i); 50 memset(match,-1,sizeof(match)); 51 for (i=1;i<=n;i++) 52 { 53 memset(vis,0,sizeof(vis)); 54 if (dfs(i)) ans++; 55 } 56 cout<<n-ans; 57 }