1-4金币阵列问题

代码是别人的,感觉写的很好。写这个博客,主要是想要重温一下思路。

枚举1~m中,每一列为第一列的情况,

//从1~n行,找出不满足的行,进行一次变换

//若是所枚举的这一列可以成功根据规则转换成目标矩阵,则,此时的矩阵与原矩阵的差别只会在列序上

此时,从i=2 列(第二列)开始与目标矩阵的第i列进行比较,

若不同,寻找本矩阵中第j列(就= i+1~m)是否有与目标矩阵的第i列相同的,若有,且 本矩阵第j列!= 目标矩阵第j列,则,进行一次列变换

//若是找不到符合条件的列,则所枚举的这一列为第一列是不可能按所给规则变换到目标矩阵的

#include<stdio.h>
const int inf = 99999;
const int N = 101;


int a[N][N],b[N][N],temp[N][N]; //a存储初始矩阵,b为目标状态矩阵
int n,m;
int need;//需要变换次数
void ChangeL(int x,int y)//变换列
{
	if(x==y)return;
	int i;
	for(i=1;i<=n;i++)
	{
		int tt=temp[i][y];
		temp[i][y]=temp[i][x];
		temp[i][x]=tt;
	}
	need++;
}
void ChangeH(int x)//变换行
{
	int i;
	for(i=1;i<=m;i++)
	{
		temp[x][i]^=1;
	}
}


bool Same(int x,int y) //判断列是否满足条件
{
	int i;
	for(i=1;i<=n;i++)
		if(b[i][x]!=temp[i][y])return false;
	return true;
}


int main()
{
	int tests;
	scanf("%d",&tests); //数据组数
	while(tests--)
	{
		scanf("%d%d",&n,&m); //n行,m列
		int i,j;
		for(i=1;i<=n;i++)
			for(j=1;j<=m;j++)
			{
				scanf("%d",&a[i][j]);
			}




			for(i=1;i<=n;i++)
				for(j=1;j<=m;j++)
					scanf("%d",&b[i][j]);


			int k;
			int ans=inf; //ans存储最终答案,初始值为无穷大


			for(k=1;k<=m;k++)//枚举各列为第一列
			{
				//copy
				for(i=1;i<=n;i++)
					for(j=1;j<=m;j++)
						temp[i][j]=a[i][j];
				need=0;
				ChangeL(1,k);


				//不满足的行,进行一次变换
				for(i=1;i<=n;i++)
				{
					if(temp[i][1]!=b[i][1])//该行不满足条件
					{
						ChangeH(i);//变换行
						need++;
					}
				}


				bool find;
				for(i=1;i<=m;i++)//检查每列是否满足条件
				{
					find=false;
					if(Same(i,i))
					{
						find=true;continue;
					}
					for(j=i+1;j<=m;j++)//寻找temp中与b的i列相同的列
					{
						if(Same(i,j))//temp 的 j列于b的i列相同
						{
							if(Same(j,j))continue;//temp的j列与b的j列相同
							ChangeL(i,j);//交换temp的i,j列
							find=true;
							break;
						}
					}
					if(find==false)//找不到该列对应列
					{
						break;
					}
				}


				if(find==true&&need<ans)ans=need;
			}


			if(ans<inf)printf("%d\n",ans);
			else
				printf("-1\n");
	}
	return 0;
}


  • 3
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 5
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值