彻底搞懂二进制枚举

当我们看到题目n 的数值 n<=12 左右时 我们就可以考虑使用二进制来进行优化了 比如 二进制枚举 dp状态压缩

二进制枚举

首先一个集合的子集有2^n个(该 n 的值一般  n<=12),所以枚举的个数有(1<<n)个;//相当 2^n

所以

for(int i=0; i<(1<<n); i++)

二进制枚举的过程如下:

每个位置值为1则保留,不为1则舍弃 ;

设s=11(二进制为1011)那么我们保留 0 1 3 位置上的数值;

那么如何找到每个位置上的数值呢?

我们遍历十进制表示(比如11),我们可以转化为二进制再枚举每一位

一个很巧妙的方式就是利用位运算;

1<<0=1(0);

1<<1=2(10);

1<<2=4(100);

1<<3=8(1000);

1<<4=16(10000);

...

1<<7=128(10000000);

我们只需要将11&(1<<i)我们便可以得到每一位是不是1 (1<<i除了那一位,剩余的都是0,所以我么就可以得到那一位是不是1)

因此,我们便有了:

for(int i=0; i<n; i++)//n位
        if(j&(1<<i))//当前状态的第i位  是否为1(存在
            printf(" %d ",a[i]);//输出 存储 或者其他操作

二进制枚举模板:

for(int i=0; i<(1<<n); i++)//二进制枚举//枚举每一个状态
    {
        for(int j=0; j<n; j++)//枚举该状态下二进制的每一位数值
        {
            if(i&(1<<j))//当前状态的第i位  是否为1(存在
                printf(" %d ",a[j]);
        }
    }

题目:ACM/ICPC 2018亚洲区预选赛北京赛站网络赛-B:Tomb Raider(模拟+二进制枚举子串)

 

  • 9
    点赞
  • 39
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值