二进制枚举

#include<stdio.h>
#include<string.h>
#include<math.h>
int m,n,tiles[15][15],tiles1[15][15],ope[15][15],a[2][5]={{-1,0,1,0,0},{0,-1,0,1,0}},res[15][15];
int reversal(int cnt)
{
    int i,j,k;
    for(i=1;i<m;i++)
    {
        for(j=0;j<n;j++)
        {
            if(tiles1[i-1][j]!=1)
                continue;
            cnt++;
            ope[i][j]=1;
            for(k=0;k<5;k++)
            {
                int oi=i+a[0][k];
                int oj=j+a[1][k];
                if(oi>=0&&oi<m&&oj>=0&&oj<n)
                {
                    tiles1[oi][oj]=tiles1[oi][oj]==1?0:1;
                }
            }
        }
        for(j=0;j<n;j++)
        {
            if(tiles1[i-1][j]==1)
                return -1;
        }
    }
    for(j=0;j<n;j++)
    {
        if(tiles1[m-1][j]==1)
            return -1;
    }
    return cnt;
}

int main()
{
    int i,j,cnt,k,ret=-1,cnto=10000;
    scanf("%d%d",&m,&n);
        for(i=0;i<m;i++)
        {
            for(j=0;j<n;j++)
            {
                scanf("%d",&tiles[i][j]);
            }
        }
        int t=1<<(n+1);//此处为找到枚举上限,因为最多为2的n次,从第一行一次都不翻到全翻,即从0到11…n…1;貌似应用longlong
        for(i=0;i<t;i++)
        {
            memset(ope,0,sizeof(ope));
            memcpy(tiles1,tiles,sizeof(tiles));
            for(cnt=j=0;j<n;j++)
            {
                ope[0][n-1-j]=i>>j&1;//对于每个翻的可能要确定每一位翻与不翻,那么要对每一位进行判断,判断他在二进制的n位数中是否这一位为一,由于只需判断这一位是否为1,可用位运算将其右移,吧要判断的移到个位,然后与一的二进制码比较,因为1除了末位是1以外全是0,这样一来进过&运算只有末位有可能考虑为1赋到ope里去,
                if(ope[0][n-1-j]==1)
                {
                    cnt++;
                    for(k=1;k<5;k++)
                    {
                        int oi=0+a[0][k];
                        int oj=n-1-j+a[1][k];
                        if(oj>=0&&oj<n)
                            tiles1[oi][oj]=tiles1[oi][oj]==1?0:1;
                    }
                }
            }
            cnt=reversal(cnt);
            if(cnt<cnto&&cnt>=0)
            {
                memcpy(res,ope,sizeof(ope));
                cnto=cnt;
            }
        }
        if(cnto==10000)
            printf("IMPOSSIBLE\n");
        else
        {
            for(i=0;i<m;i++)
            {
                for(j=0;j<n;j++)
                {
                    printf("%d%c",res[i][j],j==n-1?'\n':' ');
                }
            }
        }
    return 0;
}

主要遍历搜索周围方向的数组a不要写错,poj3279


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值