博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
POJ 3279
阅读量:6176 次
发布时间:2019-06-21

本文共 4328 字,大约阅读时间需要 14 分钟。

Time limit 2000 ms
Memory limit 65536 kB

 Description

Farmer John knows that an intellectually satisfied cow is a happy cow who will give more milk. He has arranged a brainy activity for cows in which they manipulate an M × N grid (1 ≤ M ≤ 15; 1 ≤ N ≤ 15) of square tiles, each of which is colored black on one side and white on the other side.

As one would guess, when a single white tile is flipped, it changes to black; when a single black tile is flipped, it changes to white. The cows are rewarded when they flip the tiles so that each tile has the white side face up. However, the cows have rather large hooves and when they try to flip a certain tile, they also flip all the adjacent tiles (tiles that share a full edge with the flipped tile). Since the flips are tiring, the cows want to minimize the number of flips they have to make.

Help the cows determine the minimum number of flips required, and the locations to flip to achieve that minimum. If there are multiple ways to achieve the task with the minimum amount of flips, return the one with the least lexicographical ordering in the output when considered as a string. If the task is impossible, print one line with the word "IMPOSSIBLE".

Input

Line 1: Two space-separated integers: M and N

Lines 2.. M+1: Line i+1 describes the colors (left to right) of row i of the grid with N space-separated integers which are 1 for black and 0 for white

Output

Lines 1.. M: Each line contains N space-separated integers, each specifying how many times to flip that particular location.

Sample Input

4 41 0 0 10 1 1 00 1 1 01 0 0 1

Sample Output

0 0 0 01 0 0 11 0 0 10 0 0 0

题目描述

      题目的意思是,给你一个地图,上面有白色的瓦块(用0表示)和黑色的瓦块(用1表示),现在可以对瓦块进行反转,我想了下,这不是黑白棋吗?不过这个和黑白棋还是不一样的,在这里,若对其中一块瓦块进行反转,那么它的四周的瓦块也会进行反转,题目需要我们用到最小反转次数去让整个地图的瓦片变成白色的,最终输出的是对应位置的瓦块反转的次数,而且存在多个最优解的时候,要求让由输出的图组成的字符串按逆字典序小的图输出。

      试想一下,如果反转的时候会让它的四周的瓦块同时发生反转,那么对于第一行,它不能通过只对这一行进行反转得到全白色瓦块,它只能通过第二行和第一行的反转进行结合,来让第一行的黑色瓦块变成白色瓦块;而对于最后一行的黑色瓦块,也只能通过倒数第二行和本行的结合让其中的黑色瓦块全部变成白色瓦块。

       所以我们的方法是,先将第一排的所有可能的反转情况全部逆字典序地进行枚举,这样我们就可以在同为最少步骤的时候,不需要再进行判断了,因为在同一反转次数下,逆字典序最小的先得出了。然后我们对第二行行进行枚举,若在某一列的上方,比如(2,4)和(1,4),如果上方有黑色的瓦块,那么只有通过这个位置的瓦块(此处为(2,4))进行反转,才可以让上方的瓦块变成白色。那么倒数第二行就很关键了,因为这一行造成的反转不仅要让倒数第三行的黑色瓦块变成白色瓦块,同时也要让最后一行的黑色瓦块变成白色瓦块,所以当我们利用最后一行,让倒数第二行的黑色瓦块全部变成了白色瓦块之后,如果此时最后一行的瓦块不全是白色的,说明这一操作的不成功。

       如果通过上面的操作让整个地图变成了白色的话,就说明这一反转的方法可以成功。

       为了能够输出结果,我们需要对进行反转的瓦块进行标记,由于同一块瓦片反转2次和没反转一样,所以我们规定每一块瓦块最多主动反转一次(作为中心块进行的反转),这一标记同时也代表了对地图上每一块瓦块的操作次数,可以作为答案输出,不过在此之前,我们需要统计一共进行了多少次的主动反转,然后找出最少的那一次输出,具体实现请看代码。

代码区

#include
#include
#include
#include
#include
#include
using namespace std;//对于这个数组大小,我很无奈,我取个刚刚好的长度,结果和我说不够用,哎int map[17][17]; //实际地图int choose[17][17]; //表示反转的情况int m, n;int to[5][2] = { {0,0} , {1,0} , {-1,0} , {0,1} , {0,-1} }; //表示要反转的,包括自己int solve();int main(){ scanf("%d%d", &m, &n); for(int i = 1 ;i <= m ; i ++) { for(int j = 1 ; j <= n ; j ++) { scanf("%d", *(map + i) + j); //节省时间,使用指针 } } int Min = -1; //-1代表不能反转成功,将Min设为-1,是因为我们无法确定最大值 int goal[17][17]; //存储最终方案 for(int i = 0 ; i < (1<
> (j - 1)) & 1; //记录第一行的不同的情况, int temp = solve(); //返回反转次数,但是由于调用了这一个函数,我们已经将反转的具体操作储存在choose中 if(temp >= 0 && (temp < Min || Min < 0) ) //通过异或判断可以有效地利用Min = -1 这个条件 { Min = temp; memcpy(goal, choose,sizeof(choose)); //将数据存在goal中 } } if(Min == -1) { printf("IMPOSSIBLE\n"); } else { for(int i = 1 ;i <= m ; i ++) { printf("%d", goal[i][1]); for(int j = 2; j <= n ; j ++) { printf(" %d", goal[i][j]); } cout << endl; } } return 0;}//判断坐标为(x,y)的点的颜色,用0和1表示int col(int x,int y){ int s = map[x][y]; //先记录初始颜色 for(int i = 0 ;i < 5 ; i++) { int tx = x + to[i][0]; int ty = y + to[i][1]; s += choose[tx][ty]; //choose[tx][ty]代表(tx,ty)的瓦块是否被反转过,由于颜色只用0,1表示,那么只需要加一个 1 就代表反转,加0表示不变 //而对于这个瓦块是否被反转,主要取决于他邻近的4个瓦块以及自身是否进行了反转 } return s % 2; //由于每加一代表一次颜色的反转,但是我们需要的只是0,1,所以在这里我们将之转化会用0,1表示}int solve(){ int sum = 0; //记录反转的总次数 for(int i = 2 ;i <=m ; i++) //从第二行开始枚举 { for(int j = 1 ; j <= n ; j ++) { if(col(i-1,j) == 1) //这一块的上一块是黑色的,那么就对当前这块进行反转,来让上面的那一块变成白色 { choose[i][j] = 1; //对这一块进行反转,从而让这一块上面的瓦块变成白色 sum++; //代表反转次数加一 } } } for(int i = 1 ; i <= n ; i ++) { if(col(m,i)) //若最后一行有黑色的瓦块,说明这种方法是失败的 { return -1; //失败返回-1 } sum += choose[1][i]; //顺便将第一行所用的步数加上 } return sum; //成功返回反转的次数}

 

转载于:https://www.cnblogs.com/winter-bamboo/p/10634455.html

你可能感兴趣的文章
使用编译器——Solidity中文文档(8)
查看>>
Apache用户认证、域名跳转及Apache访问日志
查看>>
POS 真的可以取代POW吗?
查看>>
Wndows下Apache+php+Mysql环境的搭建
查看>>
正则表达式
查看>>
MaxCompute安全管理指南-基础篇
查看>>
防抖和节流
查看>>
Unity3D 键盘控制物体平面移动(操作相对于摄像机方向)
查看>>
Handler 都没搞懂,拿什么去跳槽啊?
查看>>
Redis 架构演变与 Redis-cluster 群集读写方案
查看>>
keepalived的介绍及配置高可用集群
查看>>
mapreduce——实现网站PV排序两种思路
查看>>
Navicat使用教程:使用Navicat Premium 12自动执行数据库复制(一)
查看>>
IDEA报错Unable to open debugger port (127.0.0.1:51112): java.net.SocketException "socket closed"
查看>>
超详细的Python实现微博模拟登陆,小白都能懂
查看>>
遇到了平生见到最烂的代码
查看>>
Eclipse配置maven环境
查看>>
PHP正则运用
查看>>
OSChina 周四乱弹 —— 我看你TM像病毒
查看>>
中文字符和中文标点符号的正则表达式
查看>>