二进制枚举以及集合的关系运算

发布于:2023-02-14 ⋅ 阅读:(552) ⋅ 点赞:(0)

一、按位运算符

1.移位运算符

val<<shift //向左移
val>>shift //向右移

v a l 是被操作的整数值, s h i f t 是要要移动的数位 \color{orange}val是被操作的整数值,shift是要要移动的数位 val是被操作的整数值,shift是要要移动的数位

注 \color{Red}注

  • 腾出的位置用0填充,超出边界的位置被丢弃
  • 每一位是前一位的2倍,因此每左移一位相当于乘2,左移n位相当于乘2^n
  • C++左移运算符生成一个新值,不修改原来的值

2.逻辑按位运算符

①位非运算符(~),将每一位转化为它的反面(1转化而为0,0转化为1)
②或运算符 OR(|),被操作的两个值得对应位至少有一个1,则新值为1,否则为0
③与运算符 AND(&),被操作的两个值对应位都为1,则新值为1,否则为0
④异或运算符 XOR(^),一个为1,一个为0则新值为1,否则为0

注 \color{Red}注

  • 这里的位均对于二进制数来说的

二、二进制枚举

1.枚举单一集合的子集

1.1.1 相关数学概念及性质

  • 子集:如果集合A的任意一个元素都是集合B的元素,那么集合A称为集合B的子集。

  • 子集的个数:含 n 元素的子集的个数为 2 n 2^n 2n

1.1.2 集合常用关系

考虑一集合A={1,2,3,4,5}和它的四个子集A1,A2,A3,A4,如以下所示

A中元素 1 2 3 4 5 二进制 对应的十进制
在A1中情况 1 0 1 1 1 11101 29
在A2中情况 1 0 0 1 1 11001 25
在A3中情况 0 0 1 0 0 00100 4
在A4中情况 0 0 1 1 0 00110 6
  1. 并集:从元素角度来看,就是A1、A2包含的元素合并起来得到A1。

    即是按位或 :

    A1= A1|A2

  1. 交集:是指两个集合同时存在的元素组成的集合。可以看出 A1、A4的交集得到 A3。

    即是按位与:

    A3=A1&A4

  2. 属于:是指某一元素在集合中。只需检查某项元素构成的集合是否是另一集合的子集。

    一般地,可以先用左移运算构造出那个仅含一项的集合,然后再和原集合取交集 , 若不为空集,则命题成立 \color{orange}一般地,可以先用左移运算构造出那个仅含一项的集合,然后再和原集合取交集,若不为空集,则命题成立 一般地,可以先用左移运算构造出那个仅含一项的集合,然后再和原集合取交集,若不为空集,则命题成立

    • 判断第 i (1<= i <=n) 个元素是否属于某一集合 Ax
    • 即是 1<<( i-1)&Ax
  3. 补集:是指全集去掉某个集合后剩下元素组成的集合。如 A2对于全集的补集就是A3。

    即是:

    A^A2

1.1.3 如何枚举所有子集方案

对于选取一个集合中的元素构成子集,我们设 1 为 选取,0 为不选取

x1 x2 xn
0 ? 1 0 ? 1 0 ? 1 0 ? 1

由性质可知对于 n 个元素的子集不超过 2 n 2^n 2n个,于是不断缩小 n 所对应的二进制数就可以列举所有方案。

#include<iostream>

using namespace std;

int main()
{
    int n;
    
    for(int i=0;i<1<<n;i++)/* i < 2^n */
    {
        for(int j=0;j<n;j++)
            if(i>>j&1)/* 如果第 j 位 是 1 */
                cout<<1;
        	else
                cout<<0;
        
        cout<<endl;    
    }
    
    return 0;
}

注 \color{Red}注

  • 此种方法只是相对于多重循环来说,更方便,但依然是采用枚举的方式,因此只适用于枚举方案较少的情况
本文含有隐藏内容,请 开通VIP 后查看