Description
小Q是一个非常聪明的孩子,除了国际象棋,他还很喜欢玩一个电脑益智游戏——矩阵游戏。矩阵游戏在一个N
*N黑白方阵进行(如同国际象棋一般,只是颜色是随意的)。每次可以对该矩阵进行两种操作:行交换操作:选择
矩阵的任意两行,交换这两行(即交换对应格子的颜色)列交换操作:选择矩阵的任意行列,交换这两列(即交换
对应格子的颜色)游戏的目标,即通过若干次操作,使得方阵的主对角线(左上角到右下角的连线)上的格子均为黑
色。对于某些关卡,小Q百思不得其解,以致他开始怀疑这些关卡是不是根本就是无解的!!于是小Q决定写一个程
序来判断这些关卡是否有解。
Input
第一行包含一个整数T,表示数据的组数。接下来包含T组数据,每组数据第一行为一个整数N,表示方阵的大
小;接下来N行为一个N*N的01矩阵(0表示白色,1表示黑色)。
Output
输出文件应包含T行。对于每一组数据,如果该关卡有解,输出一行Yes;否则输出一行No。
Sample Input
2
2
0 0
0 1
3
0 0 1
0 1 0
1 0 0
2
0 0
0 1
3
0 0 1
0 1 0
1 0 0
Sample Output
No
Yes
【数据规模】
对于100%的数据,N ≤ 200
发现一个很神奇的事情……
就是这个矩阵如果只交换行或者只交换列无解的话
那这个矩阵就一定无解……至于为什么我也不知道
然后就可以建立二分图求最大匹配了(建行和对角线或列和对角线)
PS这道题还犯了一个沙茶错误就是二分图连边的时候顺手把反向边也连了orz
1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 using namespace std; 5 struct node 6 { 7 int to,next; 8 }edge[50001]; 9 int Ans[1010],head[1010],num_edge,used[1010],Now,T,n,x; 10 11 void add(int u,int v) 12 { 13 edge[++num_edge].to=v; 14 edge[num_edge].next=head[u]; 15 head[u]=num_edge; 16 } 17 bool find(int x) 18 { 19 for (int i=head[x];i!=0;i=edge[i].next) 20 if (used[edge[i].to]!=Now) 21 { 22 used[edge[i].to]=Now; 23 if (Ans[edge[i].to]==0 || find(Ans[edge[i].to])) 24 { 25 Ans[edge[i].to]=x; 26 return true; 27 } 28 } 29 return false; 30 } 31 32 int main() 33 { 34 scanf("%d",&T); 35 while (T--) 36 { 37 memset(head,0,sizeof(head)); 38 memset(Ans,0,sizeof(Ans)); 39 memset(used,0,sizeof(used)); 40 num_edge=0; 41 scanf("%d",&n); 42 for (int i=1;i<=n;++i) 43 for (int j=1;j<=n;++j) 44 { 45 scanf("%d",&x); 46 if (x==1) 47 add(j,i); 48 } 49 bool Flag=true; 50 for (int i=1;i<=n;++i) 51 { 52 Now=i; 53 if (!find(i)) 54 { 55 56 Flag=false; 57 break; 58 } 59 } 60 if (Flag) 61 printf("Yes "); 62 else 63 printf("No "); 64 } 65 }