Fliptile POJ-3279 DFS

本文解析了一个关于01矩阵翻转的问题,目标是最少步骤将矩阵变为全0矩阵。通过确定首行状态并递推计算后续行翻转情况,找到最优解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目链接:Fliptile

题目大意

有一个01矩阵,每一次翻转(0->1或者1->0)一个元素,就会把与他相邻的四个元素也一起翻转。求翻转哪些元素能用最少的步骤,把矩阵变成0矩阵。

思路

假设确定了第一行的状态,那么第二行需要翻转的所有元素都能确定(上一行如果是1就必须翻转)。因此对于所有第一排的情况,往下递推需要翻转的元素,最后判断最后一行如果全为0则满足条件,记录步骤最少的情况。

题解

  1 #include <iostream>
  2 #include <cstring>
  3 
  4 using namespace std;
  5 
  6 int n, m, mmin=0x3f3f3f3f;
  7 int map[20][20],tmp[20][20],ans[20][20];
  8 
  9 bool judge(){
 10     for(int i = 0 ; i < m; i++)
 11     {
 12         int time = tmp[n-2][i] + tmp[n-1][i];
 13         if(i != 0)
 14         {
 15             time += tmp[n-1][i-1];
 16         }
 17         if(i != m-1)
 18         {
 19             time += tmp[n-1][i+1];
 20         }
 21         time %= 2;
 22         if(map[n-1][i] ^ time){
 23             return false;
 24         }
 25     }
 26     return true;
 27 }
 28 
 29 void dfs(int k, int num)
 30 {
 31     if(num > mmin) return;
 32     if(k > n-1)    //最后一行遍历    结束,判断最后一行是否全0
 33     {
 34         if(judge() && num < mmin)
 35         {
 36             memcpy(ans, tmp, sizeof(tmp));
 37             mmin = num;
 38         }
 39         return;
 40     }
 41     int flag = 0;
 42     
 43     for(int i = 0; i < m; i++)
 44     {
 45         int time = 0;
 46         time = tmp[k-1][i] + tmp[k-2][i];
 47         if(i != 0)
 48         {
 49             time += tmp[k-1][i-1];
 50         }
 51         if(i != m-1)
 52         {
 53             time += tmp[k-1][i+1];
 54         }
 55         if((map[k-1][i]+time)&1)
 56         {
 57             tmp[k][i] = 1;
 58             flag++;
 59         }
 60         else
 61         {
 62             tmp[k][i] = 0;
 63         }
 64     }
 65     dfs(k+1, num+flag);        //搜索下一行
 66 }
 67 
 68 void todfs(int k, int num)
 69 {
 70     if(k > m - 1){    //遍历完第一行
 71         dfs(1, num);    //从第二行开始
 72         return;
 73     }
 74     tmp[0][k] = 0;
 75     todfs(k+1, num);
 76     tmp[0][k] = 1;
 77     todfs(k+1, num+1);
 78 }
 79 
 80 int main(int argc, char const *argv[])
 81 {
 82 #ifdef debug
 83     freopen("test.txt","r",stdin);
 84 #endif
 85     cin >> n >> m;
 86     for(int i = 0; i < n; i++)
 87     {
 88         for (int j = 0; j < m; j++)
 89         {
 90             cin >> map[i][j];
 91         }
 92     }
 93     todfs(0, 0);
 94     if(mmin == 0x3f3f3f3f)
 95     {
 96         cout << "IMPOSSIBLE" << endl;
 97     }
 98     else
 99     {
100         for(int i = 0; i < n; i++)
101         {
102             for(int j = 0; j < m; j++)
103             {
104                 cout << ans[i][j] << " ";
105             }
106             cout << endl;
107         }
108     }
109     return 0;
110 }

其实这个题用的数组下标最好是从1开始,这样不用把列号是0的元素拿出来单独计算了。

转载于:https://www.cnblogs.com/SaltyFishQF/p/10296750.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值