LG1879 「USACO2006NOV」Corn Fields 状压DP

问题描述

LG1879


题解

\(opt[i][j]\)代表前\(i\)行,且第\(i\)行状态为\(j\)的方案数。

枚举\(j\),再枚举\(k\)\(k\)为上一行的状态。

判断\(j,k\)能否共存(j&k==0

计数转移即可。

必须加强位运算能力。


\(\mathrm{Code}\)

#include<bits/stdc++.h>
using namespace std;

template <typename Tp>
void read(Tp &x){
    x=0;char ch=1;int fh;
    while(ch!='-'&&(ch>'9'||ch<'0')) ch=getchar();
    if(ch=='-') ch=getchar(),fh=-1;
    else fh=1;
    while(ch>='0'&&ch<='9') x=(x<<1)+(x<<3)+ch-'0',ch=getchar();
    x*=fh;
}

const int mod=100000000;

int ans,n,m;
int a[14];

int opt[14][(1<<12)];
bool exist[(1<<12)];
int main(){
    read(n);read(m);
    for(int i=1;i<=n;i++){
        for(int j=1,x;j<=m;j++){
            read(x);a[i]=(a[i]<<1)+x;
        }
    }
    for(int i=0;i<(1<<m);i++){
        if(((i&(i<<1))==0)&&((i&(i>>1))==0)) exist[i]=1;
    }
    opt[0][0]=1;
    for(int i=1;i<=n;i++){
        for(int j=0;j<(1<<m);j++){
            if(!exist[j]) continue;
            if((j&a[i])!=j) continue;
            for(int k=0;k<(1<<m);k++){
                if((j&k)==0) opt[i][j]=(opt[i][j]+opt[i-1][k])%mod;
            }
        }
    }
    for(int i=0;i<(1<<m);i++){
        ans=(ans+opt[n][i])%mod;
    }
    printf("%d\n",ans);
    return 0;
}
全部评论

相关推荐

11-24 00:11
已编辑
广东工业大学 算法工程师
避雷深圳&nbsp;&nbsp;yidao,试用期&nbsp;6&nbsp;个月。好嘛,试用期还没结束,就直接告诉你尽快找下一家吧,我谢谢您嘞
牛客75408465号:笑死,直属领导和 hr 口径都没统一,各自说了一些离谱的被裁理由,你们能不能认真一点呀,哈哈哈哈哈😅😅😅
点赞 评论 收藏
分享
Hello_WordN:咱就是说,除了生命其他都是小事,希望面试官平安,希望各位平时也多注意安全
点赞 评论 收藏
分享
与火:这不接? 留子的钱不挣白不挣
点赞 评论 收藏
分享
点赞 收藏 评论
分享
牛客网
牛客企业服务