程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> 一致性hash的C++實現

一致性hash的C++實現

編輯:C++入門知識

一致性哈希的C++實現

一致性哈希是分布式計算領域被廣泛應用的一個算法。在許多分布式系統包括 Amazon Dynamo, memcached, Riak 等中都有使用。

一致性哈希的原理比較簡單,網上有很多介紹的比較好的文章,也有一些相關的代碼,但是都不太令人滿意,因此自己實現了一個。代碼很簡單,放在了github上面。

consistent_hash_map

一致性哈希的功能被封裝在模板類consistent_hash_map中:

  1. template <typename T, 
  2.         typename Hash, 
  3.         typename Alloc = std::allocator<std::pair<const typename Hash::result_type,T > > > 
  4. class consistent_hash_map 

consistent_hash_map 使用了stl map風格的接口。實際上其內部也使用了std::map 來管理和維護所有節點。

consistent_hash_map只提供最基本的一致性hash的功能,並不直接支持虛擬節點的概念,但是虛擬節點的概念可以很容易的通過定 制的T 和 Hash類型來實現。這樣設計的好處在於可以使consitent_hash_map的設計和實現變得非常的簡單,同時留給用戶以極大的靈活性和可定制 性。

後面的例子中將介紹如何實現虛擬節點。

模板參數

member type

  1. size_type           Hash::reslut_type               hash函數返回值的類型 
  2. value_type          std::pair<const size_type,T>    first為節點的哈希值,second為節點 
  3. iterator            a bidirectional iterator to value_type  雙向迭代器 
  4. reverse_iterator    reverse_iterator<iterator>      反向迭代器 

member function

  1. std::size_t size() const; 
  2. 返回consistent_hash_map內的節點數量。 
  3.  
  4. bool empty() const; 
  5. 判斷consistent_hash_map是否為空 
  6.  
  7. std::pair<iterator,bool> insert(const T& node); 
  8. 插入一個節點,如果返回值中bool變量為真,iterator則為指向插入節點的迭代器。如果bool為假,表示插入失敗。 
  9. 插入失敗因為節點已經存在或者是節點的hash值與其他節點發生沖突。 
  10.  
  11. void erase(iterator it); 
  12. 通過迭代器刪除指定節點。 
  13.  
  14. std::size_t erase(const T& node); 
  15. 通過節點值刪除指定節點。 
  16.  
  17. iterator find(size_type hash); 
  18. 通過傳入的hash值找對其在consistent_hash中對應的節點的迭代器。 
  19.  
  20. iterator begin(); 
  21. iterator end(); 
  22. 返回對應迭代器 
  23.  
  24. reverse_iterator rbegin(); 
  25. reverse_iterator rend(); 
  26. 返回對應的反向迭代器 

虛擬節點的例子

整個例子的完整代碼在這。

在這個例子中,我們首先定義虛擬節點類型,和其對應的hasher。

  1. #include <stdint.h> 
  2. #include <boost/format.hpp> 
  3. #include <boost/crc.hpp> 
  4.  
  5. #include "consistent_hash_map.hpp" 
  6.  
  7. //所有主機的列表 
  8. const char* nodes[] = { 
  9.     "192.168.1.100", 
  10.     "192.168.1.101", 
  11.     "192.168.1.102", 
  12.     "192.168.1.103", 
  13.     "192.168.1.104" 
  14. }; 
  15.  
  16. //虛擬節點 
  17. struct vnode_t { 
  18.     vnode_t() {} 
  19.     vnode_t(std::size_t n,std::size_t v):node_id(n),vnode_id(v) {} 
  20.  
  21.     std::string to_str() const { 
  22.         return boost::str(boost::format("%1%-%2%") % nodes[node_id] % vnode_id); 
  23.     } 
  24.  
  25.     std::size_t node_id; //主機ID,主機在主機列表中的索引 
  26.     std::size_t vnode_id; //虛擬節點ID 
  27.  
  28. }; 
  29.  
  30. //hasher,使用CRC32作為hash算法,注意需要定義result_type 
  31. struct crc32_hasher { 
  32.     uint32_t operator()(const vnode_t& node) { 
  33.         boost::crc_32_type ret; 
  34.         std::string vnode = node.to_str(); 
  35.         ret.process_bytes(vnode.c_str(),vnode.size()); 
  36.         return ret.checksum(); 
  37.     } 
  38.     typedef uint32_t result_type; 
  39. }; 

為每個主機生成100個虛擬節點,然後加入consistent_hash_map中。

  1. typedef consistent_hash_map<vnode_t,crc32_hasher> consistent_hash_t; 
  2. consistent_hash_t consistent_hash_; 
  3.  
  4. for(std::size_t i=0;i<5;++i) { 
  5.     for(std::size_t j=0;j<100;j++) { 
  6.         consistent_hash_.insert(vnode_t(i,j)); 
  7.     } 

查找某個hash值對應的vnode 和 主機:

  1. consistent_hash_t::iterator it; 
  2. it = consistent_hash_.find(290235110); 
  3. //it -> first是該節點的hash值,it -> second是該虛擬節點。 
  4. std::cout<<boost::format("node:%1%,vnode:%2%,hash:%3%")  
  5.             % nodes[it->second.node_id] % it->second.vnode_id % it->first << std::endl; 

遍歷consistent_hash中的所有的vnode,統計每個虛擬節點的key的數量和每個主機包含key的數量:

  1. std::size_t sums[] = {0,0,0,0,0}; 
  2. consistent_hash_t::iterator i = consistent_hash_.begin(); //第一個節點 
  3. consistent_hash_t::reverse_iterator j = consistent_hash_.rbegin(); //最後一個節點 
  4. std::size_t n = i->first + UINT32_MAX - j->first; //計算第一個節點包含的key的數量 
  5. std::cout<<boost::format("vnode:%1%,hash:%2%,contains:%3%")  
  6.         % i->second.to_str() % i->first % n << std::endl; 
  7. sums[i->second.node_id] += n; //更新主機包含的key數量。 
  8.  
  9. //計算剩余所有節點包含的key的數量,並更新主機包括的key的數量。 
  10. uint32_t priv = i->first; 
  11. uint32_t cur; 
  12. consistent_hash_t::iterator end = consistent_hash_.end(); 
  13. while(++i != end) { 
  14.     cur = i->first; 
  15.     n = cur - priv; 
  16.     std::cout<<boost::format("vnode:%1%,hash:%2%,contains:%3%")  
  17.         % i->second.to_str() % cur % n << std::endl; 
  18.     sums[i->second.node_id] += n; 
  19.     priv = cur; 
  20.  
  21. for(std::size_t i=0;i<5;++i) { 
  22.     std::cout<<boost::format("node:%1% contains:%2%") % nodes[i] % sums[i] <<std::endl;  

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved