洛谷 P1924 poj 1038

Description:

  给你一个n * m的方格纸,有一些格子无法被覆盖,然后用2*3的格子覆盖这个方格纸,问你最多能放多少个格子

神级状压

为了弄清楚这道题翻了无数篇解题报告,最后终于搞明白了

用三进制表示每行的状态。

比如对于第i行第j列的格子,如果i-1行,i行的j列都是空的则用0表示,i行的j列不能放用2表示,剩下的(仅i - 1行的j列不能放)用1表示

然后深搜进行转移

干讲没意思,具体看注释,写的很清楚了(AC代码是poj的,稍微改一下输入输出就是洛谷的、

 

#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
const int N = 59050;
/*pre表示你枚举的上一个状态,now表示你现在枚举的状态*/
int dp[2][N], mp[160][15], pre[12], now[12], n, m, d, cur;
int base[] = {0,1,3,9,27,81,243,729,2187,6561,19683,59049};
void init(){
    memset(mp, 0, sizeof(mp));
    memset(dp, -1, sizeof(dp));
}
/*3进制转10进制*/
int trans(int *t){
    int ans = 0;
    for(int i = 1; i <= m; i++)
      ans += t[i] * base[i];
    return ans;
}
/*10进制转3进制*/
void chan(int t, int *p){
    for(int i = 1; i <= m; i++)
      p[i] = t % 3, t /= 3;
    return ;
}
/*i为当前行,j为当前状态,cnt为格子数目,tmp为当前压缩后整行状态*/
void dfs(int i, int j, int cnt, int tmp){
    int k;
    dp[i & 1][tmp] = max(dp[i & 1][tmp], cnt);
    if(j >= m) return ;
    /*放上3行2列的格子
      如果该层状态和前一层的状态都允许 也就是都能放格子 那么就放上格子继续搜下去*/
    if(!pre[j] && !pre[j + 1] && !now[j] && !now[j + 1]){
        now[j] = now[j + 1] = 2;  //放格子之后状态就都是2了
        k = trans(now); dfs(i, j + 2, cnt + 1, k);//因为两格都被覆盖,跳过去搜
        now[j] = now[j + 1] = 0;  //回溯
    }
    /*放上2行3列的格子
     这个就只跟你当前的状态 也就是i行j列和i-1行j列是否为空 而这一信息仅存储在now[j]上 j+1列 j+2列同理*/
    if(j < m - 1 && !now[j] && !now[j + 1] && !now[j + 2]){
        now[j] = now[j + 1] = now[j + 2] = 2; //同样修改 搜下去 回溯
        k = trans(now);  dfs(i, j + 3, cnt + 1, k);
        now[j] = now[j + 1] = now[j + 2] = 0;
    }
    dfs(i, j + 1, cnt, tmp);
    return ;
}
int main(){
    int T, x, y;
    scanf("%d", &T);
    while(T--){
        init();
        scanf("%d%d%d", &n, &m, &d);
        for(int i = 1; i <= d; i++){
            scanf("%d%d", &x, &y);
            mp[x][y] = 1;
        }
        for(int i = 1; i <= m; i++)
          pre[i] = 2;
        int tmp = trans(pre);
        dp[0][tmp] = 0;  //对于第0行来说,只有啥都不能放这一种状态 将该状态价值为0
        for(int i = 1; i <= n; i++){
            memset(dp[i & 1], -1, sizeof(dp[i & 1]));  //初始化本次要用的dp数组
            for(int j = 0; j < base[m + 1]; j++){  //枚举上一行的状态
                /* 如果上一行的状态dp数组为-1,说明上一层的该状态根本达不到,那么就不能用来转移*/
                if(dp[i + 1 & 1][j] == -1) continue;
                chan(j, pre); //将上一层的状态转换成3进制
                for(int k = 1; k <= m; k++) 
                  if(mp[i][k]) now[k] = 2; //如果此时第k位无法覆盖,那么就为状态为2
                /*不然,上层2状态变为1状态,其余为0(根据状态的定义就可以得出)*/
                  else now[k] = max(pre[k] - 1, 0); 
                cur = dp[i + 1 & 1][j];  //cur表示上一层该状态达到的最大价值
                dfs(i, 1, cur, trans(now)); //dfs转移 因为第一格前为第0格不能放,所以状态为1
            }
        }
        int ans = 0;
        for(int i = 0; i < base[m + 1]; i++)
          ans = max(ans, dp[n & 1][i]);
        printf("%d\n", ans);
    }
    return 0;
}

 

 

 

用这种多进制表示然后dfs转移的方法可以做出绝大多数的状压题,尤其是状态表示较复杂(一般需要表示两个及以上状态的),转移方程不好写而冗余状态较多的题

比如炮兵阵地,也可以用三进制表示状态然后dfs转移。但是由于炮兵阵地的总状态数并不多,所以可以直接枚举子集,写起来更省力。

posted @ 2018-07-08 02:56  Ror_shach  阅读(500)  评论(0编辑  收藏  举报