C++中的位运算和位图bitmap解析

 更新时间:2022年07月22日 16:23:46   作者:CW96  
这篇文章主要介绍了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……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表示数据存在。


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

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

确定这个数据(区间)的哪一个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++ 标准库 <chrono>的具体使用

    C++ 标准库 <chrono>的具体使用

    chrono>引入于 C++11,提供了类型安全的时间度量体系,本文就来详细的介绍一下<chrono>的使用,感兴趣的可以了解一下
    2026-02-02
  • C++基于蔡基姆拉尔森计算公式实现由年月日确定周几的方法示例

    C++基于蔡基姆拉尔森计算公式实现由年月日确定周几的方法示例

    这篇文章主要介绍了C++基于蔡基姆拉尔森计算公式实现由年月日确定周几的方法,涉及C++针对日期时间的数值运算相关操作技巧,需要的朋友可以参考下
    2017-07-07
  • 深入理解 C++ 的 std::initializer_list及使用场景分析

    深入理解 C++ 的 std::initializer_list及使用场景分析

    本文介绍了C++11引入的std::initializer_list模板类,它作为统一初始化语法的重要桥梁,封装同类型常量值,常用于容器初始化、函数参数传递和自定义类支持花括号初始化,感兴趣的朋友跟随小编一起看看吧
    2025-10-10
  • C++利用PCL点云库操作txt文件详解

    C++利用PCL点云库操作txt文件详解

    这篇文章主要为大家详细介绍了C++如何利用PCL点云库操作txt文件,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2024-01-01
  • C++编译器和链接器工作原理及使用方法完全指南

    C++编译器和链接器工作原理及使用方法完全指南

    本文将详细介绍C++中的编译器和链接器以及它们的工作原理及使用方法全面详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-05-05
  • C++中的const的使用详解

    C++中的const的使用详解

    这篇文章主要介绍了 C++中的const的使用详解的相关资料,需要的朋友可以参考下
    2017-05-05
  • C++ Boost Array与Unordered使用介绍

    C++ Boost Array与Unordered使用介绍

    Boost是为C++语言标准库提供扩展的一些C++程序库的总称。Boost库是一个可移植、提供源代码的C++库,作为标准库的后备,是C++标准化进程的开发引擎之一,是为C++语言标准库提供扩展的一些C++程序库的总称
    2022-11-11
  • C++超细致讲解队列queue的使用

    C++超细致讲解队列queue的使用

    队列先进先出,即只能在容器的末尾添加新元素,只能从头部移除元素,下面这篇文章主要给大家介绍了关于C++中队列queue用法的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2022-05-05
  • Qt私有信号实现(private signal)

    Qt私有信号实现(private signal)

    在使用Qt信号槽机制的时候,有时候我们需要一个信号只能由类内发出,而不允许使用该类对象的用户发出,此时就需要私有信号的支持,本文主要介绍了Qt私有信号实现(private signal),感兴趣的可以了解一下
    2023-10-10
  • C语言中获取和改变目录的相关函数总结

    C语言中获取和改变目录的相关函数总结

    这篇文章主要介绍了C语言中获取和改变目录的相关函数总结,包括getcwd()函数和chdir()函数以及chroot()函数的使用方法,需要的朋友可以参考下
    2015-09-09

最新评论