题意:有一个n*m的格子,每个格子都有黑白两面(0表示白色,1表示黑色)。我们需要把所有的格子都反转成黑色,每反转一个格子,它上下左右的格子都会跟着反转。请求出用最小步数完成反转时每个格子反转的次数。有多个解时,输出字典序最小的一组。

思路就是先判断第一行,然后如果第一行确定了,那么第二行就必须确定了(如果第一行某个是黑色的,那么必定要在第二行给它翻过来),所以一直到底m-1行都是确定的,接下来只要判断第m行是否是全白就好了。直接枚举第一行就可以,代码是我参考大神的代码写的,里面的二进制枚举操作还是看了有一会儿才看懂

#include <iostream>
#include <stdio.h>
#include <string.h>
using namespace std;

const int N = 20;
int gh[N][N];//初试的数组
int ch[N][N];//操作的数组
int vis[N][N];//记录翻过的牌子
int re[N][N];//
int ans[N][N];//记录答案

int fNum,aNum,n,m;

void solve( int i,int j ){
    fNum++;//操作次数+1
    vis[i][j] = 1;
    ch[i][j] = 1 - ch[i][j];
    ch[i-1][j] = 1 - ch[i-1][j];
    ch[i][j-1] = 1 - ch[i][j-1];
    ch[i+1][j] = 1 - ch[i+1][j];
    ch[i][j+1] = 1 - ch[i][j+1];
}

void record(){
    aNum = fNum;//最小次数记录一下
    for( int i = 1; i <= m; i++ ){
        for( int j = 1; j <= n; j++ ){
            ans[i][j] = vis[i][j];
        }
    }
}

int main(){
    cin >> m >> n;
    for( int i = 1; i <= m; i++ ){
        for( int j = 1; j <= n; j++ ){
            cin >> gh[i][j];
        }
    }
    aNum = 10010;
    int t = 1 << n;
    for( int loop = 0; loop < t; loop++ ){
        for( int i = 1; i <= m; i++ ){
            for( int j = 1; j <= n; j++ ){
                ch[i][j] = gh[i][j];
            }
        }
        memset(vis,0,sizeof(vis));
        fNum = 0;//记录需要操作的数目
        int moveN = loop;//这里是二进制,二进制
        for( int i = 1; i <= n; i++ ){
            if( moveN&1 ){//这里我看了很久才看明白
                //假入一个数字的二进制是0101,意思就是
                //第一格翻,第三格翻,上面t为2的N次方,包括了所有的
                //可能性的二进制对应的数字
                solve( 1,i );
            }
            moveN >>= 1;
        }
        for( int i = 2; i <= m; i++ ){
            for( int j = 1; j <= n; j++ ){
                if( ch[i-1][j] ){//如果上一行是黑色,就处理这一行
                    solve(i,j);
                }
            }
        }
        bool flag = true;
        for( int j = 1; j <= m; j++ ){
            if( ch[m][j] ){//直接判断最后一行是否满足
                flag = false;
                break;
            }
        }
        if( flag and fNum < aNum ){
            record();
        }
    }
    if( aNum == 10010 ){
        cout << "IMPOSSIBLE\n";
        return 0;
    }else{
        for( int i = 1; i <= m; i++ ){
            cout << ans[i][1];
            for( int j = 2; j <= n; j++ ){
                cout << " " << ans[i][j];
            }
            cout << endl;
        }
    }
}

poj 3279(暴力)的更多相关文章

  1. 【枚举】POJ 3279

    直达–>POJ 3279 Fliptile 题意:poj的奶牛又开始作孽了,这回他一跺脚就会让上下左右的砖块翻转(1->0 || 0->1),问你最少踩哪些砖块才能让初始的砖块全部变 ...

  2. POJ 3279(Fliptile)题解

    以防万一,题目原文和链接均附在文末.那么先是题目分析: [一句话题意] 给定长宽的黑白棋棋盘摆满棋子,每次操作可以反转一个位置和其上下左右共五个位置的棋子的颜色,求要使用最少翻转次数将所有棋子反转为黑 ...

  3. POJ 2182/暴力/BIT/线段树

    POJ 2182 暴力 /* 题意: 一个带有权值[1,n]的序列,给出每个数的前面比该数小的数的个数,当然比一个数前面比第一个数小的个数是0,省略不写,求真正的序列.(拗口) 首先想到的是从前到后暴 ...

  4. POJ 3279 Fliptile(翻格子)

    POJ 3279 Fliptile(翻格子) Time Limit: 2000MS    Memory Limit: 65536K Description - 题目描述 Farmer John kno ...

  5. POJ.3279 Fliptile (搜索+二进制枚举+开关问题)

    POJ.3279 Fliptile (搜索+二进制枚举+开关问题) 题意分析 题意大概就是给出一个map,由01组成,每次可以选取按其中某一个位置,按此位置之后,此位置及其直接相连(上下左右)的位置( ...

  6. 状态压缩+枚举 POJ 3279 Fliptile

    题目传送门 /* 题意:问最少翻转几次使得棋子都变白,输出翻转的位置 状态压缩+枚举:和之前UVA_11464差不多,枚举第一行,可以从上一行的状态知道当前是否必须翻转 */ #include < ...

  7. POJ 3279 - Fliptile - [状压+暴力枚举]

    题目链接:http://poj.org/problem?id=3279 Sample Input 4 4 1 0 0 1 0 1 1 0 0 1 1 0 1 0 0 1 Sample Output 0 ...

  8. POJ - 3279(枚举+暴力)

    Fliptile Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 14297   Accepted: 5257 Descrip ...

  9. POJ 3279 Filptile dfs

    题目链接:http://poj.org/problem?id=3279 大意:给出一块n*m的棋盘.里面放满了棋子.有1和0两种状态.给出初始状态,翻动的时候会把当前位置和当前位置的上下左右共五个位置 ...

随机推荐

  1. Servlet-知识点

    2018年10月05日 16:52:56 yigg 阅读数:38   1.JavaWeb开发的目录结构 https://blog.csdn.net/u012661010/article/details ...

  2. Java应用常用性能分析工具

    Java应用常用性能分析工具 好的工具有能有效改善和提高工作效率或加速分析问题的进度,笔者将从事Java工作中常用的性能工具和大家分享下,如果感觉有用记得投一票哦,如果你有好的工具也可以分享给我 工具 ...

  3. Android 开发 创建WiFi、WiFi热点 ---开发集合

    WIFI 权限 <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" /> < ...

  4. 反序列化失败Failed to deserialize --- local class incompatible: stream classdesc serialVersionUID

    反序列化失败: java.lang.IllegalStateException: Failed to execute CommandLineRunner at org.springframework. ...

  5. 02 while循环,密码登录

    i=3 username = "xzy" password = " while i>0: name = input("请输入你的用户名:") i ...

  6. EasyARM-iMX283A的make menuconfig出现错误:Install ncurses(ncurses-devel) and try again。

    lin@lin-machine:~/linux-2.6.35.3$ make menuconfig *** Unable to find the ncurses libraries or the ** ...

  7. virtual 函数只有在用指针或引用的方式访问,才会导致多态。

    只有用指针和引用,才会动态绑定.才会在运行时候从虚表中找对应的成员函数. 如果只是用.访问成员函数,是静态绑定,在编译时刻就固定好的. 另外,父类的虚函数,子类不管加不加virtual关键字,都是虚函 ...

  8. java 多线程学习

    一.概念 程序.进程.线程 程序   是计算机指令的集合. 进程   是一个运行中的程序,它指的是从代码加载,执行到执行结束这样一个完整过程.每个进程占用不同的内存空间. 线程   是进程中某个单一顺 ...

  9. <Dare To Dream 团队>第一次作业:团队亮相

    队名:Dare To Dream 2.团队成员组成:学号/姓名(标记团队组长): 201571030333/绽玉林(组长) 201571030132/姚慧霞 201571030308/李金平    2 ...

  10. java DES加解密及Wrong key size错误

    如下的DES加密方法会报错:Wrong key size public static String encryptDES(String source) throws Exception{ Secret ...