1.位图概念
在讲解位图之前我们先来看一道很经典的面试题。
给40亿个不重复的无符号整数,没排过序。给一个无符号整数,如何快速判断一个数是否在这40亿个数中。
我们有以下两种种解决方法:
1. 遍历,时间复杂度O(N)。(太慢)
2. 排序(O(NlogN)),利用二分查找: logN。
深入分析:解题思路2是否可行,我们先算算40亿个数据大概需要多少内存。1G=1024MB=1024*1024KB=1024*1024*1024Byte约等于10亿多Byte那么40亿个整数约等于16G,说明40亿个数是无法直接放到内存中的,只能放到硬盘文件中。而二分查找只能对内存数组中的有序数据进行查找。
其实我们并不需要开如此大的空间,我们只需要知道这个数据在或者不在,刚好是两种状态,那么可以使用一个二进制比特位来代表数据是否存在的信息,如果二进制比特位为1,代表存在,为0代表不存在。
位图概念
所谓位图,就是用每一位来存放某种状态,适用于海量数据,数据无重复的场景。通常是用来判断某个数据存不存在的。
2.位图的实现
namespace bit
{
template<size_t N>
class bitset
{
public:
private:
std::vector<int> _bs;
};
}
现在我们要做的是如何修改x 对应的位置比特位。(如下)
i = x / 32 得到 x在vector中的第 i 个整形数据。
a = x % 32 得到 x在第 i 位的第 a位。
namespace bit
{
template<size_t N>
class bitset
{
public:
bitset()
{
_bs.resize(N / 32 + 1);
}
void set(size_t x)//表示x 存在
{
int i = x / 32;
int a = x % 32;
_bs[i] |= (1 << a);
}
void reset(size_t x)//表示x 不存在
{
int i = x / 32;
int a = x % 32;
_bs[i] &= (~(1 << a));
}
//x处映射的值是1 返回真
//x处映射的值是0 返回假
bool rest(size_t x)
{
int i = x / 32;
int a = x % 32;
return _bs[i] & (1 << a);
}
private:
std::vector<int> _bs;
};
}
int main()
{
bit::bitset<100> bs;
bs.set(32);
bs.reset(32);
bs.set(33);
cout<<bs.rest(31)<<endl;
cout<<bs.rest(32)<<endl;
cout<<bs.rest(33)<<endl;
cout<<bs.rest(34)<<endl;
cout<<bs.rest(35)<<endl;
return 0;
}
测试结果如下:
3.bitset的使用
如果要开大量空间我们可以这样
但是库里面是开不出来的
因为我们自己实现的底层使用的是vector 开空间是在堆上面,而库里面底层实现是静态数组,所以开不出来。
可以看到我们无论开多大的空间他本身所占的内存都是16因为我们的核心空间都开在堆上的。
库里面就不一样了如下:
大家使用库里面bitset的就要小心使用。
解决办法如下:
位图的优缺点:
- 优点:增删查改快,节省空间
- 缺点:只能适用整形
接下来看几道关于位图的题目:
- 给定40亿个整数,设计算法找到只出现一次的整数。
我们可以用
00:表示0次
01:表示1次
10:表示2次及以上
void testbitset(vector<int>& arr)
{
bit::bitset<-1> bs1;
bit::bitset<-1> bs2;
for (auto& e : arr)
{
if (!bs1.rest(e) && !bs2.rest(e))//00->01
{
bs2.set(e);
}
else if(!bs1.rest(e) && bs2.rest(e))//01->10
{
bs2.reset(e);
bs1.set(e);//10
}
else//10->11
{
bs2.set(e);
}
}
for (int e = 0; e < 100; e++)
{
if (!bs1.rest(e) && bs2.rest(e))
{
//出现一次
cout << "1->:" << e << endl;
}
else if (bs1.rest(e) && !bs2.rest(e))
{
//出现两次
cout << "2->:" << e << endl;
}
else if(bs1.rest(e) && bs2.rest(e))
{
//出现三次及以上
cout << "3->:" << e << endl;
}
}
}
int main()
{
vector<int> v1({0, 0, 1, 1, 2, 2, 2, 3, 4, 4, 5, 5, 6, 6, 6, 6});
testbitset(v1);
}
感谢大家的观看!