• <位图(bitset)和布隆过滤器(BloomFilter)>——《C++高阶》


    目录

    1. 哈希的应用

    1.1 位图

    1.1.1 位图概念

    1.1.2 位图的实现

    1.1.3 位图的应用

    1.2 布隆过滤器

    1.2.1 布隆过滤器提出

    1.2.2布隆过滤器概念

     1.2.3 布隆过滤器的插入

    1.2.4 布隆过滤器的查找

    1.2.5 布隆过滤器删除

    1.2.6 布隆过滤器优点

    1.2.7 布隆过滤器缺陷

    2. 海量数据处理类题目解析

    2.1 哈希切割

    2.2 位图应用

    2.3 布隆过滤器

    3.位图与布隆过滤器完整源码:

    3.1位图模拟实现源码:

    3.2 布隆过滤器模拟实现源码:

    3.3 附用的unordered_set、unordered_map、OpenHash:

    3.4 test.cpp:​

    后记:●由于作者水平有限,文章难免存在谬误之处,敬请读者斧正,俚语成篇,恳望指教!

                                                                               ——By 作者:新晓·故知


    1. 哈希的应用

    1.1 位图

    1.1.1 位图概念

    1. 面试题

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

    这40亿个数中。【腾讯】

    分析:不能使用set/unordered_set,因为内存不够。也不能使用外排序+二分查找,因为不能很好的支持快速随机访问!

    1. 遍历,时间复杂度O(N)

    2. 排序(O(NlogN)),利用二分查找: logN

    3. 位图解决

    数据是否在给定的整形数据中,结果是在或者不在,刚好是两种状态,那么可以使用一

    个二进制比特位来代表数据是否存在的信息,如果二进制比特位为1,代表存在,为0

    代表不存在。比如:

    直接定址法哈希——用一个比特位标识映射值在不在!

    2. 位图概念

    所谓位图,就是用每一位来存放某种状态,适用于海量数据,数据无重复的场景。通常是用

    来判断某个数据存不存在的。

    1.1.2 位图的实现

    1. class bitset
    2. {
    3. public:
    4. bitset(size_t bitCount)
    5. : _bit((bitCount>>5)+1), _bitCount(bitCount)
    6. {}
    7. // 将which比特位置1
    8. void set(size_t which)
    9. {
    10. if(which > _bitCount)
    11. return;
    12. size_t index = (which >> 5);
    13. size_t pos = which % 32;
    14. _bit[index] |= (1 << pos);
    15. }
    16. // 将which比特位置0
    17. void reset(size_t which)
    18. {
    19. if(which > _bitCount)
    20. return;
    21. size_t index = (which >> 5);
    22. size_t pos = which % 32;
    23. _bit[index] &= ~(1<
    24. }
    25. // 检测位图中which是否为1
    26. bool test(size_t which)
    27. {
    28. if(which > _bitCount)
    29. return false;
    30. size_t index = (which >> 5);
    31. size_t pos = which % 32;
    32. return _bit[index] & (1<
    33. }
    34. // 获取位图中比特位的总个数
    35. size_t size()const{ return _bitCount;}
    36. // 位图中比特为1的个数
    37. size_t Count()const
    38. {
    39.     int bitCnttable[256] = {
    40. 0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8};
    41. size_t size = _bit.size();
    42. size_t count = 0;
    43. for(size_t i = 0; i < size; ++i)
    44. {
    45. int value = _bit[i];
    46. int j = 0;
    47. while(j < sizeof(_bit[0]))
    48. {
    49. unsigned char c = value;
    50. count += bitCntTable[c];
    51. ++j;
    52. value >>= 8;
    53. }
    54. }
    55. return count;
    56. }
    57. private:
    58. vector<int> _bit;
    59. size_t _bitCount;
    60. };

    1.1.3 位图的应用

    1. 快速查找某个数据是否在一个集合中

    2. 排序 + 去重

    3. 求两个集合的交集、并集等

    4. 操作系统中磁盘块标记

    1.2 布隆过滤器

    1.2.1 布隆过滤器提出

    我们在使用新闻客户端看新闻时,它会给我们不停地推荐新的内容,它每次推荐时要去重,去掉那些已经看过的内容。问题来了,新闻客户端推荐系统如何实现推送去重的? 用服务器记录了用户看过的所有历史记录,当推荐系统推荐新闻时会从每个用户的历史记录里进行筛选,过滤掉那些已经存在的记录。 如何快速查找呢?

    1. 用哈希表存储用户记录,缺点:浪费空间

    2. 用位图存储用户记录,缺点:位图一般只能处理整形,如果内容编号是字符串,就无法处理了。

    3. 将哈希与位图结合,即布隆过滤器

    1.2.2布隆过滤器概念

    布隆过滤器是由布隆(Burton Howard Bloom)在1970年提出的 一种紧凑型的、比较巧妙的率型数据结构,特点是高效地插入和查询,可以用来告诉你 “某样东西一定不存在或者可能存在”,它是用多个哈希函数,将一个数据映射到位图结构中。此种方式不仅可以提升查询效率,也可以节省大量的内存空间

     1.2.3 布隆过滤器的插入

    1. struct BKDRHash
    2. {
    3. size_t operator()(const string& s)
    4. {
    5. // BKDR
    6. size_t value = 0;
    7. for (auto ch : s)
    8. {
    9. value *= 31;
    10. value += ch;
    11. }
    12. return value;
    13. }
    14. };
    15. struct APHash
    16. {
    17. size_t operator()(const string& s)
    18. {
    19. size_t hash = 0;
    20. for (long i = 0; i < s.size(); i++)
    21. {
    22. if ((i & 1) == 0)
    23. {
    24. hash ^= ((hash << 7) ^ s[i] ^ (hash >> 3));
    25. }
    26. else
    27. {
    28. hash ^= (~((hash << 11) ^ s[i] ^ (hash >> 5)));
    29. }
    30. }
    31. return hash;
    32. }
    33. };
    34. struct DJBHash
    35. {
    36. size_t operator()(const string& s)
    37. {
    38. size_t hash = 5381;
    39. for (auto ch : s)
    40. {
    41. hash += (hash << 5) + ch;
    42. }
    43. return hash;
    44. }
    45. };
    46. template<size_t N,
    47. size_t X = 5,
    48. class K = string,
    49. class HashFunc1 = BKDRHash,
    50. class HashFunc2 = APHash,
    51. class HashFunc3 = DJBHash>
    52. class BloomFilter
    53. {
    54. public:
    55. void Set(const K& key)
    56. {
    57. size_t len = X*N;
    58. size_t index1 = HashFunc1()(key) % len;
    59. size_t index2 = HashFunc2()(key) % len;
    60. size_t index3 = HashFunc3()(key) % len;
    61. /* cout << index1 << endl;
    62. cout << index2 << endl;
    63. cout << index3 << endl<
    64. _bs.set(index1);
    65. _bs.set(index2);
    66. _bs.set(index3);
    67. }
    68. bool Test(const K& key)
    69. {
    70. size_t len = X*N;
    71. size_t index1 = HashFunc1()(key) % len;
    72. if (_bs.test(index1) == false)
    73. return false;
    74. size_t index2 = HashFunc2()(key) % len;
    75. if (_bs.test(index2) == false)
    76. return false;
    77. size_t index3 = HashFunc3()(key) % len;
    78. if (_bs.test(index3) == false)
    79. return false;
    80. return true;  // 存在误判的
    81. }
    82. // 不支持删除,删除可能会影响其他值。
    83. void Reset(const K& key);
    84. private:
    85. bitset _bs;
    86. };

    1.2.4 布隆过滤器的查找

    布隆过滤器的思想是将一个元素用多个哈希函数映射到一个位图中,因此被映射到的位置的比特

    位一定为1。所以可以按照以下方式进行查找:分别计算每个哈希值对应的比特位置存储的是否为零,只要有一个为零,代表该元素一定不在哈希表中,否则可能在哈希表中

    注意:布隆过滤器如果说某个元素不存在时,该元素一定不存在,如果该元素存在时,该元素可能存在,因为有些哈希函数存在一定的误判。

    比如:在布隆过滤器中查找"alibaba"时,假设3个哈希函数计算的哈希值为:1、3、7,刚好和其他元素的比特位重叠,此时布隆过滤器告诉该元素存在,但实该元素是不存在的。

    1.2.5 布隆过滤器删除

    布隆过滤器不能直接支持删除工作,因为在删除一个元素时,可能会影响其他元素。

    比如:删除上图中"tencent"元素,如果直接将该元素所对应的二进制比特位置0,“baidu”元素也被删除了,因为这两个元素在多个哈希函数计算出的比特位上刚好有重叠。

    一种支持删除的方法:将布隆过滤器中的每个比特位扩展成一个小的计数器,插入元素时给k个计数器(k个哈希函数计算出的哈希地址)加一,删除元素时,给k个计数器减一,通过多占用几倍存储空间的代价来增加删除操作。

    缺陷:

    1. 无法确认元素是否真正在布隆过滤器中

    2. 存在计数回绕

    1.2.6 布隆过滤器优点

    1. 增加和查询元素的时间复杂度为:O(K), (K为哈希函数的个数,一般比较小),与数据量大小无关

    2. 哈希函数相互之间没有关系,方便硬件并行运算

    3. 布隆过滤器不需要存储元素本身,在某些对保密要求比较严格的场合有很大优势

    4. 在能够承受一定的误判时,布隆过滤器比其他数据结构有这很大的空间优势

    5. 数据量很大时,布隆过滤器可以表示全集,其他数据结构不能

    6. 使用同一组散列函数的布隆过滤器可以进行交、并、差运算

    1.2.7 布隆过滤器缺陷

    1. 有误判率,即存在假阳性(False Position),即不能准确判断元素是否在集合中(补救方法:再建立一个白名单,存储可能会误判的数据)

    2. 不能获取元素本身

    3. 一般情况下不能从布隆过滤器中删除元素

    4. 如果采用计数方式删除,可能会存在计数回绕问题

    2. 海量数据处理类题目解析

    2.1 哈希切割

    给一个超过100G大小的log fifile, log中存着IP地址, 设计算法找到出现次数最多的IP地址?

    与上题条件相同,如何找到top K的IP?如何直接用Linux系统命令实现?

    2.2 位图应用

    1. 给定100亿个整数,设计算法找到只出现一次的整数?

    分析:使用两个比特位,标识3种状态。出现0次:00  出现1次:01  出现2次及以上:10

    2. 给两个文件,分别有100亿个整数,我们只有1G内存,如何找到两个文件交集?

    3. 位图应用变形:1个文件有100亿个int,1G内存,设计算法找到出现次数不超过2次的所有整数

    注:放在位图找交集的意义是先去重。

     位图的优点:

    1.节省空间

    2.处理快速

    缺点:只能针对整型映射

    2.3 布隆过滤器

    1. 给两个文件,分别有100亿个query,我们只有1G内存,如何找到两个文件交集?分别给出

    精确算法和近似算法

    2. 如何扩展BloomFilter使得它支持删除元素的操作

    布隆过滤器映射的位数越多,冲突的概率越小

    布隆过滤器的底层就是一个位图
    布隆过滤器支持删除,但删除会影响其他位值,因此会付出很大的代价。删除的方式是计数。

     

    3.位图与布隆过滤器完整源码:

    因为涉及到附用unordered_set、unordered_map,所以这里给出完整的源码。

    3.1位图模拟实现源码:

    bitset:

    位图1:

    1. #pragma once
    2. #include
    3. //位图1
    4. namespace my
    5. {
    6. // N个比特位的位图 10 16
    7. template<size_t N>
    8. class bitset
    9. {
    10. public:
    11. bitset()
    12. {
    13. // +1保证足够比特位,最多浪费8个
    14. _bits.resize(N / 8 + 1, 0);
    15. }
    16. //x映射的位标记成1
    17. void set(size_t x)
    18. {
    19. // x映射的比特位在第几个char对象
    20. size_t i = x / 8;
    21. // x在char第几个比特位
    22. size_t j = x % 8;
    23. _bits[i] |= (1 << j);
    24. }
    25. void reset(size_t x)
    26. {
    27. // x映射的比特位在第几个char对象
    28. size_t i = x / 8;
    29. // x在char第几个比特位
    30. size_t j = x % 8;
    31. //! && ||
    32. //~ & |
    33. _bits[i] &= (~(1 << j));
    34. }
    35. bool test(size_t x)
    36. {
    37. // x映射的比特位在第几个char对象
    38. size_t i = x / 8;
    39. // x在char第几个比特位
    40. size_t j = x % 8;
    41. return _bits[i] & (1 << j);
    42. }
    43. private:
    44. std::vector<char> _bits;
    45. //vector _bits;
    46. };
    47. template<size_t N>
    48. class two_bitset
    49. {
    50. public:
    51. void set(size_t x)
    52. {
    53. int in1 = _bs1.test(x);
    54. int in2 = _bs2.test(x);
    55. if (in1 == 0 && in2 == 0)
    56. {
    57. _bs2.set(x);
    58. }
    59. else if (in1 == 0 && in2 == 1)
    60. {
    61. _bs1.set(x);
    62. _bs2.reset(x);
    63. }
    64. }
    65. bool is_once(size_t x)
    66. {
    67. return _bs1.test(x) == 0 && _bs2.test(x) == 1;
    68. }
    69. private:
    70. bitset _bs1;
    71. bitset _bs2;
    72. };
    73. }

     位图2:

    1. #pragma once
    2. #include
    3. //位图2
    4. namespace my
    5. {
    6. // N个比特位的位图 10 16
    7. template<size_t N>
    8. class bitset
    9. {
    10. public:
    11. bitset()
    12. {
    13. // +1保证足够比特位,最多浪费8个
    14. _bits.resize(N / 8 + 1, 0);
    15. }
    16. //x映射的位标记成1
    17. void set(size_t x)
    18. {
    19. // x映射的比特位在第几个char对象
    20. size_t i = x / 8;
    21. // x在char第几个比特位
    22. size_t j = x % 8;
    23. _bits[i] |= (1 << j);
    24. }
    25. void reset(size_t x)
    26. {
    27. // x映射的比特位在第几个char对象
    28. size_t i = x / 8;
    29. // x在char第几个比特位
    30. size_t j = x % 8;
    31. //! && ||
    32. //~ & |
    33. _bits[i] &= (~(1 << j));
    34. }
    35. bool test(size_t x)
    36. {
    37. // x映射的比特位在第几个char对象
    38. size_t i = x / 8;
    39. // x在char第几个比特位
    40. size_t j = x % 8;
    41. return _bits[i] & (1 << j);
    42. }
    43. private:
    44. std::vector<char> _bits;
    45. //vector _bits;
    46. };
    47. template<size_t N>
    48. class two_bitset
    49. {
    50. public:
    51. void set(size_t x)
    52. {
    53. int in1 = _bs1.test(x);
    54. int in2 = _bs2.test(x);
    55. //00
    56. if (in1 == 0 && in2 == 0)
    57. {
    58. _bs2.set(x); //01
    59. }
    60. //01
    61. else if (in1 == 0 && in2 == 1)
    62. {
    63. _bs1.set(x); //10
    64. _bs2.reset(x);
    65. }
    66. //10
    67. else if (in1 == 1 && in2 == 0)
    68. {
    69. _bs2.set(x); //11
    70. }
    71. }
    72. bool is_once(size_t x)
    73. {
    74. return _bs1.test(x) == 0 && _bs2.test(x) == 1;
    75. }
    76. private:
    77. bitset _bs1;
    78. bitset _bs2;
    79. };
    80. }

    3.2 布隆过滤器模拟实现源码:

    BloomFilter:

    3.3 附用的unordered_set、unordered_map、OpenHash:

    unordered_set:

    1. #include"OpenHT.h"
    2. namespace my
    3. {
    4. template<class K, class HashFunc = DefaultHash>
    5. class unordered_set
    6. {
    7. //内部类
    8. struct SetKeyOfT
    9. {
    10. const K& operator()(const K& key)
    11. {
    12. return key;
    13. }
    14. };
    15. public:
    16. typedef typename Bucket::HashTable::iterator iterator;
    17. iterator begin()
    18. {
    19. return _ht.begin();
    20. }
    21. iterator end()
    22. {
    23. return _ht.end();
    24. }
    25. //bool insert(const K& key)
    26. pairbool> insert(const K& key)
    27. {
    28. return _ht.Insert(key);
    29. }
    30. iterator find(const K& key)
    31. {
    32. return _ht.Find(key);
    33. }
    34. bool erase(const K& key)
    35. {
    36. return _ht.Erase(key);
    37. }
    38. private:
    39. Bucket::HashTable _ht;
    40. };
    41. }

    unordered_map:

    1. #include"OpenHT.h"
    2. namespace my
    3. {
    4. template<class K, class V, class HashFunc = DefaultHash>
    5. class unordered_map
    6. {
    7. //内部类
    8. struct MapKeyOfT
    9. {
    10. const K& operator()(const pair& kv)
    11. {
    12. return kv.first;
    13. }
    14. };
    15. public:
    16. //定义map的迭代器
    17. typedef typename Bucket::HashTable, MapKeyOfT, HashFunc> ::iterator iterator;
    18. iterator begin()
    19. {
    20. return _ht.begin();
    21. }
    22. iterator end()
    23. {
    24. return _ht.end();
    25. }
    26. //bool insert(const pair& kv)
    27. pairbool> insert(const pair& kv)
    28. {
    29. return _ht.Insert(kv);
    30. }
    31. iterator find(const K& key)
    32. {
    33. return _ht.Find(key);
    34. }
    35. bool erase(const K& key)
    36. {
    37. return _ht.Erase(key);
    38. }
    39. V& operator[](const K& key)
    40. {
    41. //pair ret = insert(make_pair(key, V()));
    42. auto ret = insert(make_pair(key, V()));
    43. return ret.first->second; //first是迭代器,而map里面访问second要用->,其实这里有两个->,但为了可读性
    44. }
    45. private:
    46. Bucket::HashTable, MapKeyOfT,HashFunc> _ht;
    47. };
    48. }

    OpenHash:

    1. #pragma once
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. using namespace std;
    10. //仿函数
    11. template<class K>
    12. struct DefaultHash //1.普通类直接强转
    13. {
    14. size_t operator()(const K& key)
    15. {
    16. return(size_t)key; //支持取模,强转为整数
    17. }
    18. };
    19. template<>
    20. struct DefaultHash //String类特化
    21. {
    22. size_t operator()(const string& key)
    23. {
    24. //4.BKDR法
    25. size_t hash = 0;
    26. for (auto ch : key)
    27. {
    28. hash = hash * 131 + ch;
    29. }
    30. return hash;
    31. }
    32. };
    33. namespace Bucket
    34. {
    35. template<class T>
    36. struct HashNode
    37. {
    38. T _data;
    39. HashNode* _next;
    40. HashNode(const T& data)
    41. :_data(data)
    42. , _next(nullptr)
    43. {}
    44. };
    45. template<class K, class T, class KeyOfT, class HashFunc>
    46. class HashTable;
    47. template<class K, class T, class KeyOfT, class HashFunc>
    48. class __HTIterator //单向迭代器
    49. {
    50. typedef HashNode Node;
    51. typedef __HTIterator Self;
    52. public:
    53. Node* _node;
    54. HashTable* _pht;
    55. __HTIterator(){} //默认哈希迭代器构造函数
    56. __HTIterator(Node* node, HashTable* pht)
    57. :_node(node)
    58. ,_pht(pht)
    59. {}
    60. Self& operator++()
    61. {
    62. if (_node->_next)
    63. {
    64. _node = _node->_next;
    65. }
    66. else
    67. {
    68. KeyOfT kot;
    69. HashFunc hf;
    70. size_t hashi = hf(kot(_node->_data))%_pht->_tables.size();
    71. ++hashi;
    72. //找下一个不为空的桶
    73. for (; hashi < _pht->_tables.size(); ++hashi)
    74. {
    75. if (_pht->_tables[hashi])
    76. {
    77. _node = _pht->_tables[hashi];
    78. break;
    79. }
    80. }
    81. // 没有找到不为空的桶,用nullptr去做end标识
    82. if (hashi == _pht->_tables.size())
    83. {
    84. _node = nullptr;
    85. }
    86. }
    87. return *this;
    88. }
    89. T& operator*()
    90. {
    91. return _node->_data;
    92. }
    93. T* operator->()
    94. {
    95. return &_node->_data;
    96. }
    97. bool operator!=(const Self& s) const
    98. {
    99. return _node != s._node;
    100. }
    101. bool operator==(const Self& s) const
    102. {
    103. return _node == s._node;
    104. }
    105. };
    106. //unordered_set--->HashTable _ht;
    107. //unordered_map--->HashTable, MapKeyOfT> _ht;
    108. template<class K, class T, class KeyOfT, class HashFunc>
    109. class HashTable
    110. {
    111. template<class K, class T, class KeyOfT, class HashFunc>
    112. friend class __HTIterator;
    113. typedef HashNode Node;
    114. public:
    115. typedef __HTIterator iterator;
    116. iterator begin()
    117. {
    118. for (size_t i = 0; i < _tables.size(); ++i)
    119. {
    120. Node* cur = _tables[i];
    121. if (cur)
    122. {
    123. return iterator(cur, this);
    124. }
    125. }
    126. return end();
    127. }
    128. iterator end()
    129. {
    130. return iterator(nullptr, this);
    131. }
    132. //手动实现Node的析构函数
    133. ~HashTable()
    134. {
    135. for (size_t i = 0; i < _tables.size(); ++i)
    136. {
    137. Node* cur = _tables[i];
    138. while (cur)
    139. {
    140. Node* next = cur->_next;
    141. delete cur;
    142. cur = next;
    143. }
    144. _tables[i] = nullptr;
    145. }
    146. }
    147. ///
    148. HashTable() = default; //强制生成默认构造函数
    149. 链表桶的深拷贝
    150. //HashTable(const HashTable ht)
    151. //{
    152. // int sz = ht._tables.size();
    153. // for (int i = 0; i < sz; i++)
    154. // {
    155. // Node* cur = ht._tables[i];
    156. // while (cur)
    157. // {
    158. // this->Insert(cur->_kv);
    159. // cur = cur->_next;
    160. // }
    161. // }
    162. // _n = ht._n;
    163. //}
    164. 链表桶的赋值重载
    165. //HashTable& operator=(HashTable ht)
    166. //{
    167. // //现代写法
    168. // ht._tablle.swap(_tables);
    169. // _n = ht._n;
    170. //}
    171. ///
    172. size_t GetNextPrime(size_t prime)
    173. {
    174. const int PRIMECOUNT = 28;
    175. static const size_t primeList[PRIMECOUNT] =
    176. {
    177. 53ul, 97ul, 193ul, 389ul, 769ul,
    178. 1543ul, 3079ul, 6151ul, 12289ul, 24593ul,
    179. 49157ul, 98317ul, 196613ul, 393241ul, 786433ul,
    180. 1572869ul, 3145739ul, 6291469ul, 12582917ul, 25165843ul,
    181. 50331653ul, 100663319ul, 201326611ul, 402653189ul, 805306457ul,
    182. 1610612741ul, 3221225473ul, 4294967291ul
    183. };
    184. // 获取比prime大那一个素数
    185. size_t i = 0;
    186. for (; i < PRIMECOUNT; ++i)
    187. {
    188. if (primeList[i] > prime)
    189. return primeList[i];
    190. }
    191. return primeList[i];
    192. }
    193. //bool Insert(const T& data)
    194. pairbool> Insert(const T& data)
    195. {
    196. HashFunc hf;
    197. KeyOfT kot;
    198. //if (Find(kot(data))) //使用仿函数,不知道data是set还是map
    199. //if (Find(kot(data))!=end()) //使用仿函数,不知道data是set还是map
    200. iterator pos = Find(kot(data));
    201. if(pos!=end())
    202. {
    203. //return false;
    204. return make_pair(pos,false);
    205. }
    206. //负载因子(这里设置为1) 扩容
    207. if (_tables.size() == _n)
    208. {
    209. //写法2:扩容
    210. //size_t newSize = _tables.size() == 0 ? 10 : _tables.size() * 2;
    211. //写法3:
    212. size_t newSize = GetNextPrime(_tables.size());
    213. if (newSize != _tables.size())
    214. {
    215. vector newTable;
    216. newTable.resize(newSize, nullptr);
    217. for (size_t i = 0; i < _tables.size(); ++i)
    218. {
    219. Node* cur = _tables[i];
    220. while (cur)
    221. {
    222. size_t hashi = hf(kot(cur->_data)) % newSize; //hf转换为整型,kot()取k
    223. cur->_next = newTable[hashi];
    224. newTable[hashi] = cur;
    225. cur = cur->_next;
    226. }
    227. _tables[i] = nullptr;
    228. }
    229. newTable.swap(_tables);
    230. }
    231. }
    232. size_t hashi = hf(kot(data)); //两层仿函数调用
    233. hashi %= _tables.size();
    234. //头插到对应的桶即可
    235. Node* newnode = new Node(data);
    236. newnode->_next = _tables[hashi];
    237. _tables[hashi] = newnode;
    238. ++_n;
    239. //return true;
    240. return make_pair(iterator(newnode,this),false);
    241. }
    242. //Node* Find(const K& key)
    243. iterator Find(const K& key)
    244. {
    245. if (_tables.size() == 0)
    246. {
    247. //return nullptr;
    248. return iterator(nullptr,this);
    249. }
    250. KeyOfT kot;
    251. //写法1:有名对象
    252. HashFunc hf;
    253. size_t hashi = hf(key);
    254. //写法2:匿名对象
    255. //size_t hashi = HashFunc()(key);
    256. hashi %= _tables.size();
    257. Node* cur = _tables[hashi];
    258. while (cur)
    259. {
    260. if (kot(cur->_data) == key)
    261. {
    262. //return cur;
    263. return iterator(cur, this);
    264. }
    265. cur = cur->_next;
    266. }
    267. //return nullptr; //效率高,是因为直接返回了指针
    268. return iterator(nullptr, this);
    269. }
    270. bool Erase(const K& key)
    271. {
    272. if (_tables.size() == 0)
    273. {
    274. return false;
    275. }
    276. KeyOfT kot;
    277. //写法1:有名对象
    278. HashFunc hf;
    279. size_t hashi = hf(key);
    280. hashi %= _tables.size();
    281. Node* cur = _tables[hashi];
    282. Node* prev = nullptr;
    283. while (cur)
    284. {
    285. if (kot(cur->_data) == key)
    286. {
    287. if (prev == nullptr)
    288. {
    289. _tables[hashi] = cur->_next;
    290. }
    291. else
    292. {
    293. prev->_next = cur->_next;
    294. }
    295. delete cur;
    296. return true;
    297. }
    298. cur = cur->_next;
    299. }
    300. return false;
    301. }
    302. private:
    303. //指针数组
    304. //没有使用list,是因为list是双向链表,成本大 vector> _tables; 但不用手动写析构函数
    305. vector _tables;
    306. //开散列采用挂起,如果新表扩容,那么当旧表释放,vector会将自己的释放,
    307. //但是挂在vector的结点Node* 不会自动释放,因为Node* 是内置类型,需要手动释放。
    308. size_t _n = 0;
    309. };
    310. }

    3.4 test.cpp:

    1. #include"OpenHT.h"
    2. #include"UnorderedSet.h"
    3. #include"UnorderedMap.h"
    4. #include"BitSet.h"
    5. #include"BloomFilter.h"
    6. //
    7. //1.库中的UnorderedSet、UnorderedMap测试
    8. void test_std_UnorderedSet()
    9. {
    10. std::unordered_set<int> s;
    11. s.insert(9);
    12. s.insert(7);
    13. s.insert(2);
    14. s.insert(1);
    15. s.insert(6);
    16. s.insert(5);
    17. //1.迭代器遍历
    18. //unordered_set::iterator it = s.begin();
    19. //auto it = s.begin();
    20. //写法2:
    21. std::unordered_set<int>::iterator it;
    22. it = s.begin();
    23. while (it != s.end())
    24. {
    25. cout << *it << " ";
    26. ++it;
    27. }
    28. cout << endl;
    29. //2.范围for遍历
    30. for (auto e : s)
    31. {
    32. cout << e << " ";
    33. }
    34. cout << endl;
    35. }
    36. void test_std_UnorderedMap()
    37. {
    38. std::unordered_map dict;
    39. dict.insert(make_pair("sort", "排序"));
    40. dict.insert(make_pair("left", "左边"));
    41. dict.insert(make_pair("left", "剩余"));
    42. dict["string"];
    43. dict["left"] = "剩余";
    44. dict["string"] = "字符串";
    45. //迭代器遍历
    46. //std::unordered_map::iterator it = dict.begin();
    47. auto it = dict.begin();
    48. while (it != dict.end())
    49. {
    50. cout << it->first << " " << it->second << endl;
    51. ++it;
    52. }
    53. cout << endl;
    54. //范围for遍历
    55. for (auto& kv : dict)
    56. {
    57. cout << kv.first << " " << kv.second << endl;
    58. }
    59. }
    60. //
    61. //2.模拟实现UnorderedSet、UnorderedMap测试
    62. void test_my_UnorderedSet()
    63. {
    64. my::unordered_set<int> s;
    65. s.insert(9);
    66. s.insert(7);
    67. s.insert(2);
    68. s.insert(1);
    69. s.insert(6);
    70. s.insert(5);
    71. //1.迭代器遍历
    72. //my::unordered_set::iterator it = s.begin();
    73. //auto it = s.begin();
    74. //写法2:
    75. my::unordered_set<int>::iterator it;
    76. it = s.begin();
    77. while (it != s.end())
    78. {
    79. cout << *it << " ";
    80. ++it;
    81. }
    82. cout << endl;
    83. //2.范围for遍历
    84. for (auto e : s)
    85. {
    86. cout << e << " ";
    87. }
    88. cout << endl;
    89. }
    90. void test_my_UnorderedMap()
    91. {
    92. my::unordered_map dict;
    93. dict.insert(make_pair("sort", "排序"));
    94. dict.insert(make_pair("left", "左边"));
    95. dict.insert(make_pair("left", "剩余"));
    96. dict["string"];
    97. dict["left"] = "剩余";
    98. dict["string"] = "字符串";
    99. //迭代器遍历
    100. //my::unordered_map::iterator it = dict.begin();
    101. auto it = dict.begin();
    102. while (it != dict.end())
    103. {
    104. cout << it->first << " " << it->second << endl;
    105. ++it;
    106. }
    107. cout << endl;
    108. //范围for遍历
    109. for (auto& kv : dict)
    110. {
    111. cout << kv.first << " " << kv.second << endl;
    112. }
    113. }
    114. //
    115. //3.模拟实现bitset测试
    116. void test_my_bit_set()
    117. {
    118. my::bitset<100> bs;
    119. bs.set(18);
    120. bs.set(19);
    121. cout << bs.test(18) << endl;
    122. cout << bs.test(19) << endl;
    123. cout << bs.test(20) << endl;
    124. bs.reset(18);
    125. bs.reset(18);
    126. cout << bs.test(18) << endl;
    127. cout << bs.test(19) << endl;
    128. cout << bs.test(20) << endl;
    129. //my::bitset bigBS;
    130. my::bitset<0xFFFFFFFF> bigBS;
    131. //my::bitset<-1> bigBS;
    132. //bit_set<2^32-1> bigBS; // pow
    133. }
    134. //4.模拟实现two_bitset
    135. void test_my_two_bitset()
    136. {
    137. int a[] = { 4, 3, 2, 4, 5, 2, 2, 4, 7, 8, 9, 2, 1,3,7 };
    138. my::two_bitset<10> tbs;
    139. for (auto e : a)
    140. {
    141. tbs.set(e);
    142. }
    143. for (size_t i = 0; i < 10; ++i)
    144. {
    145. if (tbs.is_once(i))
    146. {
    147. cout << i << " ";
    148. }
    149. }
    150. cout << endl;
    151. }
    152. //5.模拟实现BloomFilter测试
    153. void test_my_BloomFilter1()
    154. {
    155. //插入10个数据
    156. //写法1:43是计算的值(具体参见文档)
    157. //my::BloomFilter bf;
    158. //写法2:43是计算的值(具体参见文档),因为模板给了缺省值,因此这里只传第一个数据类型即可
    159. my::BloomFilter<43> bf;
    160. string a[] = { "铭记历史","勿忘国耻","振兴中华","吾辈自强","1931.09.18","九*一八事变","2022.09.18"};
    161. for (auto e : a)
    162. {
    163. cout << e <<" ";
    164. }
    165. cout << endl;
    166. for (auto& e : a)
    167. {
    168. bf.Set(e);
    169. }
    170. for (auto& e : a)
    171. {
    172. cout << bf.Test(e) << endl;
    173. }
    174. cout << bf.Test("缅怀先烈") << endl;
    175. cout << bf.Test("吾辈自强") << endl;
    176. cout << bf.Test("砥砺前行") << endl;
    177. }
    178. //6.模拟实现布隆过滤器测试对比误判率
    179. void test_my_BloomFilter2()
    180. {
    181. srand(time(0));
    182. //给不同的值,观察误判率
    183. //一般来说,空间越大,冲突率越小
    184. const size_t N = 10000;
    185. /*const size_t N = 60000;
    186. const size_t N = 100000;*/
    187. /*my::BloomFilter<5 * N> bf;
    188. my::BloomFilter<6 * N> bf;*/
    189. my::BloomFilter<8 * N> bf;
    190. std::vector v1;
    191. std::string url = "https://www.cnblogs.com/-clq/archive/2012/05/31/2528153.html";
    192. for (size_t i = 0; i < N; ++i)
    193. {
    194. v1.push_back(url + std::to_string(1234 + i));
    195. }
    196. for (auto& str : v1)
    197. {
    198. bf.Set(str);
    199. }
    200. //打印判断的key的bool返回值(1或0)
    201. /*for (auto& str : v1)
    202. {
    203. cout << bf.Test(str) << endl;
    204. }
    205. cout << endl << endl;*/
    206. //给一个相似的字符串
    207. std::vector v2;
    208. for (size_t i = 0; i < N; ++i)
    209. {
    210. std::string url = "https://www.cnblogs.com/-clq/archive/2012/05/31/2528153.html";
    211. url += std::to_string(999999 + i);
    212. v2.push_back(url);
    213. }
    214. //判断并统计。如果在,就是误判
    215. size_t n2 = 0;
    216. for (auto& str : v2)
    217. {
    218. if (bf.Test(str))
    219. {
    220. ++n2;
    221. }
    222. }
    223. cout << "相似字符串误判率:" << (double)n2 / (double)N << endl;
    224. std::vector v3;
    225. for (size_t i = 0; i < N; ++i)
    226. {
    227. string url = "zhihu.com";
    228. url += std::to_string(rand());
    229. v3.push_back(url);
    230. }
    231. size_t n3 = 0;
    232. for (auto& str : v3)
    233. {
    234. if (bf.Test(str))
    235. {
    236. ++n3;
    237. }
    238. }
    239. cout << "不相似字符串误判率:" << (double)n3 / (double)N << endl;
    240. }
    241. int main()
    242. {
    243. //test_std_UnorderedSet();
    244. //test_std_UnorderedMap();
    245. //test_my_UnorderedSet();
    246. //test_my_UnorderedMap();
    247. //test_my_bit_set();
    248. //test_my_two_bitset();
    249. //test_my_BloomFilter1();
    250. test_my_BloomFilter2();
    251. return 0;
    252. }

    后记:
    ●由于作者水平有限,文章难免存在谬误之处,敬请读者斧正,俚语成篇,恳望指教!

                                                                               ——By 作者:新晓·故知

  • 相关阅读:
    某些设置由你的组织来管理
    知名外企嵌入式C语言笔试试题
    微服务中feign远程调用相关的各种超时问题
    a股level2数据接口的最优委托信息
    Jenkins 打包到其他服务器上执行
    影刀rpa办公实例一,汇总报名表
    证件照处理
    WiFi蓝牙模块促进传统零售数字化转型:智能零售体验再升级
    QT调用python程序出现问题Failed to get function
    常见React Hooks 钩子函数用法
  • 原文地址:https://blog.csdn.net/m0_57859086/article/details/126862124