• 位运算简介以及常用技巧


    A.什么是位运算 ?

    计算机里的内存都是用 二进制 储存的,说白了位运算就是对这些 二进制数 去操作
    由于是直接对 二进制数 去进行操作,就会有许多优秀的性质.
    一般来说有这么几个常用的 位运算 符号 :

    位运算符号 名称 规则 例子
    & 与运算符 相同位的两个数字都为 (1) 则为 (1); 若有一个不为 (1), 则为 (0) 1100 & 1010 = 1000
    | 或运算符 相同位要有一个为 1 则为 1 1100 | 1010 = 1110
    ^ 异或运算符 相同位不同则为 (1), 相同则为 (0) 1100 ^ 1010 = 0110
    ~ 取反运算符 (0)(1) 全部取反 ~1001 = 0110
    << 左移运算 x << n 相当与 (x imes 2 ^ n) 101 << 2 = 10100
    >> 右移运算 x >> n 相当于 (x / 2 ^ n) 1010 >> 2 = 10

    B.二进制枚举

    0. 代替一类 0/1 dfs

    一个二进制数的 (0/1) 可以代表集合中某一个元素选与不选
    既然这样我们只要枚举一个二进制数就可以代替递归版本的 dfs
    这样做虽然可以避免递归操作的较大常数
    但是我们只得到了一个二进制数,我们还要把它用 (O(logn)) 的时间复杂度把它解码,emmm...反正看情况哪个方便用哪个呗qwq

    1. 枚举子集

    for(int i = S; i; i = i - 1 & S)
    

    2. 枚举包含指定元素的集合

    for(int i = S; i; i = i + 1 | S)
    

    3. 枚举指定个数元素的集合*(常数有点大)

    int x, b, t, c, m, r;
    	x = BinRd();
    	b = x & -x;
    	t = x + b; 
    	c = t ^ x; 
    	m = (c >> 2) / b;
    	r = t | m;
    

    C.一种位运算的前缀求和

    1.求:

    [sum_{i = 1}^nsum_{j =1}^iA_i xor A_j ]

    可以去处理数组 (pre[i][32][0 / 1]) 表示 (i) 之前的数某一位上有多少个 (0 / 1)

    每一位分别 (logn) 计算答案,(logn) 修改 前缀 (pre) 数组

    假如把 (xor) 换成 (and) 的话只要计算某一位上有多少个 (1) 就好了,可以少一维

    2.求:

    [sum_{i < j < k} (A_ixorA_j) imes (A_jxorA_k) ]

    发现 (j) 在中间,那么我们求一个 (bit[32][0 / 1]) 前缀和,求一个 (bit[32][0 / 1]) 后缀和

    枚举每一个位置 (logn) 计算就好了

  • 相关阅读:
    ManualResetEvent详解
    MEF搜索范围
    ThreadStart和ParameterizedThreadStart区别
    快速理解C#高级概念(一) Delegate委托
    c# 多线程 --Mutex(互斥锁)
    sql server 得到数据库字典
    遇到的坑:在线用户统计的实现
    asp.net core 使用EF7 Code First 创建数据库,同时使用命令创建数据库
    使用cachemanager做缓存(Session的缓存)
    C# web api返回类型设置为json的两种方法
  • 原文地址:https://www.cnblogs.com/Lskkkno1/p/11420783.html
Copyright © 2020-2023  润新知