poj2441 Arrange the Bulls (状态压缩dp)

问题描述

有N只牛和M个谷仓(N和M小于等于20)
每只牛只喜欢在特定的谷仓里面
给出每只牛喜欢的谷仓
问所有牛都在自己喜欢的谷仓里的方案数

输入

在输入的第一行中包含两个整数N和M (1 <= N <= 20, 1 <= M <= 20)。然后是N行。第i行首先包含一个整数P (1 <= P <= M),它表示我喜欢在其中玩耍的谷仓奶牛的数量。然后是P个整数,它给出了P个谷仓的数量。

输出

在一行中打印一个整数,这是解决方案的数量。(保证答案在int范围内)

分析:

1表示谷仓有牛了,0表示谷仓没有牛

用d[ j ]表示j状态下的方案数,状态 j 的二进制中1的个数就是该状态下牛的数量
放牛的顺序是不影响结果的,不妨从第一头牛开始。

轮到插入第i头牛的时候
遍历所有状态,找到其中只有(i-1)头牛的状态(即找到二进制数中1的个数为(i-1)的状态)
判断这个状态下能否插入当前牛(遍历当前牛喜欢的每一个谷仓),如果可以,则
d[插入之后的状态]+=d[插入之前的状态]

另外0头牛的时候方案数为1,即d[0]=1;

code:
#include<iostream>
#include<cstdio>
#include<vector>
using namespace std;
const int maxm=23;
int n,m;
vector<int>like[maxm];//存放每只牛的可行位置
int d[1<<maxm];
int one(int x){//统计二进制下1的个数
    int ans=0;while(x){x&=(x-1);ans++;}return ans;
}
int main(){
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++){
        int k;
        scanf("%d",&k);
        while(k--){
            int t;
            scanf("%d",&t);
            like[i].push_back(1<<(t-1));//可行位置
        }
    }
    d[0]=1;
    for(int i=1;i<=n;i++){//放第i头牛
        for(int j=0;j<(1<<m);j++){//所有状态
            if(one(j)!=(i-1))continue;//如果1的个数不是i-1个就跳过
            for(int k=0;k<(int)like[i].size();k++){//遍历当前牛喜欢的所有谷仓的位置
                if((like[i][k]&j)==0){//如果能插入
                    d[like[i][k]|j]+=d[j];//之后的状态用或运算|  求出
                }
            }
        }
    }
    int ans=0;
    for(int j=1;j<(1<<m);j++){//统计答案
        if(one(j)==n){
            ans+=d[j];
        }
    }
    printf("%d\n",ans);
    return 0;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值