neuq-acm预备队训练week 10 P1129 [ZJOI2007] 矩阵游戏

题目描述

小 Q 是一个非常聪明的孩子,除了国际象棋,他还很喜欢玩一个电脑益智游戏――矩阵游戏。矩阵游戏在一个 n×n 黑白方阵进行(如同国际象棋一般,只是颜色是随意的)。每次可以对该矩阵进行两种操作:

  • 行交换操作:选择矩阵的任意两行,交换这两行(即交换对应格子的颜色)。
  • 列交换操作:选择矩阵的任意两列,交换这两列(即交换对应格子的颜色)。

游戏的目标,即通过若干次操作,使得方阵的主对角线(左上角到右下角的连线)上的格子均为黑色。

对于某些关卡,小 Q 百思不得其解,以致他开始怀疑这些关卡是不是根本就是无解的!于是小 Q 决定写一个程序来判断这些关卡是否有解。

输入格式

本题单测试点内有多组数据

第一行包含一个整数 T,表示数据的组数,对于每组数据,输入格式如下:

第一行为一个整数,代表方阵的大小 n。 接下来 n 行,每行 n 个非零即一的整数,代表该方阵。其中 0 表示白色,1 表示黑色。

输出格式

对于每组数据,输出一行一个字符串,若关卡有解则输出 Yes,否则输出 No

输入输出样例

解题思路

匈牙利二分图匹配

AC代码

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
int n,m,T,ans,x,vis[200010],A[200010];
vector<int> g[200010];
bool find(int x);
int main ( )
{
  cin>>T;
  while(T--)
    {
  	cin>>n;
  	ans=0;
  	for(int i=1;i<=n;i++)
  		g[i].clear(),A[i] = 0;
  	for(int i=1;i<=n;i++)
  		for(int j=1;j<=n;j++)
  		{
  			scanf("%d",&x);
  			if(x)
  				g[i].push_back(j);
  		}
  	for(int i = 1 ; i <= n ;i++)
  	{
  		memset(vis , 0 ,sizeof(vis));
  		ans += find(i);
  	}
  	puts(ans==n?"Yes":"No");
  }
}

bool find(int x)
{
  for(int i=0;i<g[x].size();i++)
    {
  	if(!vis[g[x][i]])
  	{
  		vis[g[x][i]] = 1;
  		if(!A[g[x][i]] || find(A[g[x][i]]))
  		{
  			A[g[x][i]]=x;
  			return 1;
  		}
  	}
  }
  return 0;
}
相关推荐
爱吃生蚝的于勒1 小时前
C语言内存函数
c语言·开发语言·数据结构·c++·学习·算法
phoenix@Capricornus3 小时前
循环矩阵和BCCB矩阵与向量乘积的快速计算——矩阵向量乘积与频域乘积之间的转换
线性代数·矩阵
神仙别闹6 小时前
基本MFC类框架的俄罗斯方块游戏
c++·游戏·mfc
ChoSeitaku6 小时前
链表循环及差集相关算法题|判断循环双链表是否对称|两循环单链表合并成循环链表|使双向循环链表有序|单循环链表改双向循环链表|两链表的差集(C)
c语言·算法·链表
Fuxiao___6 小时前
不使用递归的决策树生成算法
算法
我爱工作&工作love我7 小时前
1435:【例题3】曲线 一本通 代替三分
c++·算法
白-胖-子7 小时前
【蓝桥等考C++真题】蓝桥杯等级考试C++组第13级L13真题原题(含答案)-统计数字
开发语言·c++·算法·蓝桥杯·等考·13级
workflower7 小时前
数据结构练习题和答案
数据结构·算法·链表·线性回归
好睡凯7 小时前
c++写一个死锁并且自己解锁
开发语言·c++·算法
Sunyanhui17 小时前
力扣 二叉树的直径-543
算法·leetcode·职场和发展