zoukankan      html  css  js  c++  java
  • 位运算简介以及常用技巧

    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) 计算就好了

  • 相关阅读:
    《Real Time 3D Terrain Engines Using C++ And DirectX9》(实时地形引擎)随书源码
    (转)ogreUML类图
    (转)导弹跟踪算法
    转:正则表达式语法
    读取数据
    Python lambda用法及其与def的区别
    转Vb6.0安装失败解决办法,完全卸载VB(提高班里win7没装上VB的可以看看,我实验成功了)
    vb imagelist 作用
    二叉树的实现
    转:Python语言编程学习资料(电子书+视频教程)下载汇总:
  • 原文地址:https://www.cnblogs.com/Lskkkno1/p/11420783.html
Copyright © 2011-2022 走看看