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 小时前
【论文导读】ORB-SLAM3:An Accurate Open-Source Library for Visual, Visual-Inertial and
深度学习·算法
骑自行车的码农3 小时前
【React用到的一些算法】游标和栈
算法·react.js
博笙困了3 小时前
AcWing学习——双指针算法
c++·算法
moonlifesudo3 小时前
322:零钱兑换(三种方法)
算法
NAGNIP21 小时前
大模型框架性能优化策略:延迟、吞吐量与成本权衡
算法
美团技术团队1 天前
LongCat-Flash:如何使用 SGLang 部署美团 Agentic 模型
人工智能·算法
摸鱼的春哥1 天前
10年3次大失败,他从“罪人”输成了中年人的“白月光”
游戏
Fanxt_Ja1 天前
【LeetCode】算法详解#15 ---环形链表II
数据结构·算法·leetcode·链表
侃侃_天下1 天前
最终的信号类
开发语言·c++·算法
茉莉玫瑰花茶1 天前
算法 --- 字符串
算法