温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

C++中的位运算和位图bitmap实例分析

发布时间:2022-07-28 11:13:36 来源:亿速云 阅读:96 作者:iii 栏目:开发技术

本篇内容介绍了“C++中的位运算和位图bitmap实例分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

    位运算总结

    移位运算

    • 移位运算是双目运算符,两个运算分量都是整形,结果也是整形。

    • “<<” 左移:右边空出的位上补0,左边的位将从首位挤掉,其值相当于乘2。

    • ">>"右移:右边的位被挤掉。对于左边移出的空位,如果是正数则空位补0,若为负数,可能补0或补1,这取决于所用的计算机系统。

    二进制补码运算公式:

    -x = ~x + 1 = ~(x-1)
    -(~x) = x+1
    ~(-x) = x-1
    x+y = x - ~y - 1 = (x|y)+(x&y)
    x-y = x + ~y + 1 = (x|~y)-(~x&y)
    x^y = (x|y)-(x&y)
    x|y = (x&~y)+y
    x&y = (~x|y)-~x
    x==y:    ~(x-y|y-x)
    x!=y:    x-y|y-x
    x< y:    (x-y)^((x^y)&((x-y)^x))
    x<=y:    (x|~y)&((x^y)|~(y-x))
    x< y:    (~x&y)|((~x|y)&(x-y))//无符号x,y比较
    x<=y:    (~x|y)&((x^y)|~(y-x))//无符号x,y比较

    位运算应用举例

    (1) 判断int型变量a是奇数还是偶数

    a&1 = 0 偶数 
    a&1 = 1 奇数

    (2) 取int型变量a的第k位 (k=0,1,2&hellip;&hellip;sizeof(int)),即a>>k&1

    (3) 将int型变量a的第k位清0,即

    a = a&~(1<<k)

    (4) 将int型变量a的第k位置1,

    a=a|(1<<k)

    (5) int型变量循环左移k次,

    a=a<<k|a>>sizeof(unsigned int)*8-k

    (6) int型变量a循环右移k次,

    a=a>>k|a<<sizeof(unsigned int)*8-k

    (7) 整数的平均值

    对于两个整数x,y,如果用 (x+y)/2 求平均值,会产生溢出,因为 x+y 可能会大于INT_MAX,但是我们知道它们的平均值是肯定不会溢出的,我们用如下算法:

    int average(int x, int y)   //返回X,Y 的平均值
    {   
         return (x&y)+((x^y)>>1);
    }

    (8)判断一个整数是不是2的幂,对于一个数 x >= 0,判断他是不是2的幂

    bool power2(int x)
    {
        return ((x&(x-1))==0)&&(x!=0);
    }

    (9)不用 temp交换两个整数,可以是负整数

    void swap( int& x , int& y)
    {
        x ^= y;
        y ^= x;
        x ^= y;
    }
    
    void swap01(int& x , int& y){
       x += y;
       y = x - y;
       x = x - y;
    }

    (10) 计算绝对值

    int abs( int x )
    {
    int y ;
    y = x >> 31 ;
    return (x^y)-y ;        //or: (x+y)^y
    }
    
    int abs01(int a){
    	return (a>0)?a:(~a+1);
    }

    (11) 取模运算转化成位运算 (在不产生溢出的情况下)

    a % (2^n) 等价于 a & (2^n - 1)

    (12)乘法运算转化成位运算 (在不产生溢出的情况下)

    a * (2^n) 等价于 a<< n

    (13)除法运算转化成位运算 (在不产生溢出的情况下)

      a / (2^n) 等价于 a>> n
        例: 12/8 == 12>>3

    (14) a % 2 等价于 a & 1

    (15) x 的 相反数 表示为 (~x+1)

    (16)两整数相加,可以是负整数

    int add(int a,int b){
    	while(b!=0){
    		int temp=a^b;
    		b=(unsigned int)(a&b)<<1;
    		a = temp;
    	}
    	return a;
    }

    位图

    题目

    给40亿个不重复的无符号整数,没排过序。给一个无符号整数,如何快 速判断一个数是否在这40亿个数中。 【腾讯】

    思路

    这道题首先要判断40亿个不重复的无符号整数究竟占多大的内存,因为太大的内存我们无法加载到现有的计算机中。

    一个整数是4个字节,40亿个整数就是160亿个字节,也就相当于16G内存,就一般的计算机而言很难实现这个加载,所以我们可以采取以下两种方案,一种是分割,一种是位图。

    方法

    ①分割

    采用分割处理,把40亿个数分批次处理完毕,当然可以实现我们最终的目标,但是这样做时间复杂度未免优点太高。

    ②位图BitMap

    在介绍这种方法前我首先来介绍一下什么是位图。

    位图BitMap:位图是一个数组的每一个数据的每一个二进制位表示一个数据,0表示数据不存在,1表示数据存在。


    C++中的位运算和位图bitmap实例分析

    如上所示,当我们需要存放一个数据的时候,我们需要安装以下方法:

    首先确定这个数字在整个数据的哪一个数据(区间)。

    确定这个数据(区间)的哪一个Bit位上。

    在这个位上置1即可。

    实现代码:

    #include <iostream>
    #include <vector>
    using namespace std;
    
    class BitMap
    {
    public:
        BitMap(size_t range)
        {
            //此时多开辟一个空间
            _bits.resize(range / 32 + 1);
        }
        void Set(size_t x)
        {
            int index = x / 32;//确定哪个数据(区间)
            int temp = x % 32;//确定哪个Bit位
            _bits[index] |= (1 << temp);//位操作即可
        }
        void Reset(size_t x)
        {
            int index = x / 32;
            int temp = x % 32;
            _bits[index] &= ~(1 << temp);//取反
        }
        bool Test(size_t x)
        {
            int index = x / 32;
            int temp = x % 32;
            if (_bits[index]&(1<<temp))
                return 1;
            else
                return 0;
        }
    
    private:
        vector<int> _bits;
    };
    
    void TestBitMap()
    {
        BitMap am(-1);
        BitMap bm(200);
        bm.Set(136);
        bm.Set(1);
        cout << bm.Test(136) << endl;
        bm.Reset(136);
        cout << bm.Test(136) << endl;
    }
    
    int main()
    {
        TestBitMap();
        return 0;
    }

    “C++中的位运算和位图bitmap实例分析”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注亿速云网站,小编将为大家输出更多高质量的实用文章!

    向AI问一下细节

    免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

    AI