c语言:金币阵列的问题

  有m*n(m <=100,n <=100)个金币在桌面上排成一个m行n 列的金币阵列。每一枚金币或正面朝上或背面朝上。用数字表示金币状态,0表示金币正面朝上,1 表示背面朝上。

  金币阵列游戏的规则是: (1)每次可将任一行金币翻过来放在原来的位置上;

  (2)每次可任选2 列,交换这2 列金币的位置。

   编程任务:给定金币阵列的初始状态和目标状态,编程计算按金币游戏规则,将金币阵列从初始状态变换到目标状态所需的最少变换次数。

Input

  输入数据有多组数据。第1行有1 个正整数k,表示有k 组数据。每组数据的第1 行有2 个正整数m 和n。以下的m行是金币阵列的初始状态,每行有n 个数字表示该行金币的状态,0 表示金币正面朝上,1 表示背面朝上。接着的m行是金币阵列的目标状态。

Output

  将计算出的最少变换次数按照输入数据的次序输出。相应数据无解时输出-1。

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

枚举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++)//枚举各列为第一列
             {
                 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;
 }

(0)

相关推荐

  • C语言金币阵列问题解决方法

    本文实例详细讲述了C语言实现金币阵列问题的解决方法,分享给大家供大家参考.具体方法如下: 问题描述: 有m*n(1 ≤ m, n ≤ 100)个金币在桌面上排成一个 m 行 n 列的阵列.每一枚金币或正面朝上或背面朝上.用数字表示金币状态,0表示金币正面朝上,1 表示背面朝上. 金币阵列游戏的规则是: 1. 每次可将任一行金币翻过来放在原来的位置上: 2. 每次可任选 2 列,交换这 2 列金币的位置. 本题要求对于给定的金币阵列初始状态和目标状态,编程计算按金币游戏规则,将金币阵列从初始状态变

  • c语言:金币阵列的问题

    有m*n(m <=100,n <=100)个金币在桌面上排成一个m行n 列的金币阵列.每一枚金币或正面朝上或背面朝上.用数字表示金币状态,0表示金币正面朝上,1 表示背面朝上. 金币阵列游戏的规则是: (1)每次可将任一行金币翻过来放在原来的位置上: (2)每次可任选2 列,交换这2 列金币的位置. 编程任务:给定金币阵列的初始状态和目标状态,编程计算按金币游戏规则,将金币阵列从初始状态变换到目标状态所需的最少变换次数. Input 输入数据有多组数据.第1行有1 个正整数k,表示有k 组数据

  • 利用C语言玩转魔方阵实例教程

    魔方阵 魔方阵,古代又称"纵横图",是指组成元素为自然数1.2-n的平方的n×n的方阵,其中每个元素值都不相等,且每行.每列以及主.副对角线上各n个元素之和都相等. 如3×3的魔方阵: 8 1 6 3 5 7 4 9 2 魔方阵的排列规律如下: (1)将1放在第一行中间一列: (2)从2开始直到n×n止各数依次按下列规则存放:每一个数存放的行比前一个数的行数减1,列数加1(例如上面的三阶魔方阵,5在4的上一行后一列): (3)如果上一个数的行数为1,则下一个数的行数为n(指最下一行);

  • 基于Go语言实现分金币游戏

    目录 问题 代码实现 运行效果 问题 你有50枚金币,需要分配给以下几个人:Matthew,Sarah,Augustus,Heidi,Emilie,Peter,Giana,Adriano,Aaron,Elizabeth. 分配规则如下: a. 名字中每包含'e'或'E'分1枚金币 b. 名字中每包含'i'或'I'分2枚金币 c. 名字中每包含'o'或'O'分3枚金币 d: 名字中每包含'u'或'U'分4枚金币 写一个程序,计算每个用户分到多少金币,以及最后剩余多少金币? 程序结构如下,请实现 ‘

  • 易语言制作王者荣耀刷金币脚本的代码

    打开黑夜模拟器,按下F10,王者荣耀进入挑战-魔女回忆,开始即可. 王者荣耀刷金币脚本 此功能需要加载精易模块5.6 .版本 2 .支持库 shellEx .支持库 EThread .支持库 eAPI .程序集 窗口程序集_启动窗口 .程序集变量 热键F10, 整数型 .程序集变量 热键Home, 整数型 .程序集变量 集_线程句柄, 整数型 .子程序 __启动窗口_创建完毕 热键F10 = 注册热键 (取窗口句柄 (), 标签1.取窗口句柄 (), 0, #F10键) 热键Home = 注册热

  • 易语言写GTA金币修改软件

    本教程来使用"易语言"修改游戏中的金钱. 1.运行"ce". 2.运行"gta-sa". 3.在任意的目录中新建一个易语言源码. (我是在桌面新建的) 4.在"gta-sa"中玩新的游戏,初始有350美元. 5.在"CE"窗口中选择"选择进程",然后选择一个名为"gta_sa.exe"的进程. 6.在"ce"的窗口中"Hex"字

  • 必须知道的C语言八大排序算法(收藏)

    概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的排序记录,在排序过程中需要访问外存. 我们这里说说八大排序就是内部排序. 当n较大,则应采用时间复杂度为O(nlog2n)的排序方法:快速排序.堆排序或归并排序序. 快速排序:是目前基于比较的内部排序中被认为是最好的方法,当待排序的关键字是随机分布时,快速排序的平均时间最短: 1.插入排序-直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到

  • c语言解析bmp图片的实例

    心血来潮想了解下常用图片的格式解析,翻看了一些资料后,发现最简单的是bmp格式,所以先拿它开刀. BMP格式 这种格式内的数据分为三到四个部分,依次是: 文件信息头 (14字节)存储着文件类型,文件大小等信息 图片信息头 (40字节)存储着图像的尺寸,颜色索引,位平面数等信息 调色板 (由颜色索引数决定)[可以没有此信息] 位图数据 (由图像尺寸决定)每一个像素的信息在这里存储 一般的bmp图像都是24位,也就是真彩.每8位为一字节,24位也就是使用三字节来存储每一个像素的信息,三个字节对应存放

  • Java语言求解完美数代码分析

    1.概念 首先我们理解一下,什么叫做完美数? 问题描述:若一个自然数,它所有的真因子(即除了自身以外的约数)的和恰好等于它本身,这种数叫做完全数.简称"完数" 例如, 6=1+2+3 28=1+2+4+7+14 496=1+2+4+8+16+31+62+124+248 8128=1+2+4+8+16+32+64+127+254+508+1016+2032+4064 按照完数的定义,其实用程序求解完数并不是太难,先求解出这个数的所有真因子,然后相加,判断是否等于它本身即可.但是,在这个数

  • 用Python写王者荣耀刷金币脚本

    王者荣耀很多朋友都想买脚本和挂之类的,想更加容易的获得金币等可以在游戏里买英雄等,今天我们发挥程序员的优势教给大家用Python语言自己写一个可以刷金币的脚本,以下是全部内容. 王者荣耀的冒险模式里有个挑战模式,第一次过关可以获得比较多的金币,后面重新挑战还是会获得少量金币,这不算是bug,只有你不嫌烦手动蛮力也可以刷金币. 推荐关卡:陨落的废都 - 魔女回忆 此关卡使用纯输出英雄20秒左右可以打BOSS,50秒左右可以通关,每次重复通关可以获得奖励19金币.在开挂前建议你手动通关体验一下.此为

随机推荐