本篇内容介绍了“bitcoin中的limitedmap有什么用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!
bitcoin 里面的limitedmap 基本上是在std::map 上实现了优先队列的功能, 主要用在记录向对等节点索取inv 对应的时间记录。 以前不了解, stl 容器可以如此灵活使用, 把多个容器组合一起,完成更强大的功能。limitedmap 的主要技巧是,使用一个multimap(limitedmap 中多个不同的key可能映射到同一个value), 记录map里面value及其在map的位置(迭代器), bitcoin 里面有不少这种用法, 一个容器的元素是另一个容器的迭代器。
template <typename K, typename V> class limitedmap { public: typedef K key_type; typedef V mapped_type; typedef std::pair<const key_type, mapped_type> value_type; typedef typename std::map<K, V>::const_iterator const_iterator; typedef typename std::map<K, V>::size_type size_type; protected: std::map<K, V> map; typedef typename std::map<K, V>::iterator iterator; std::multimap<V, iterator> rmap; typedef typename std::multimap<V, iterator>::iterator rmap_iterator; size_type nMaxSize; public: // 构造器建立class 的不变量,元素size 不超过nMaxSizeIn explicit limitedmap(size_type nMaxSizeIn) { assert(nMaxSizeIn > 0); nMaxSize = nMaxSizeIn; } //迭代器,只读方法都代理给底层的std::map const_iterator begin() const { return map.begin(); } const_iterator end() const { return map.end(); } size_type size() const { return map.size(); } bool empty() const { return map.empty(); } const_iterator find(const key_type& k) const { return map.find(k); } size_type count(const key_type& k) const { return map.count(k); } //这里插入采用了先插入新元素, 然后再检测个数超限,超限后再删除最小的元素(破坏了不变量,然后再恢复回来) //为什么不插入前,先检测个数是否超限, 已经超限就返回,不用再后来删除了 //每次往底层map成功地添加一个元素,就更新multimap, 记录新的值在map中的位置 void insert(const value_type& x) { std::pair<iterator, bool> ret = map.insert(x); if (ret.second) { if (map.size() > nMaxSize) { map.erase(rmap.begin()->second); rmap.erase(rmap.begin()); } rmap.insert(make_pair(x.second, ret.first)); } } //按key删除对应的元素,首先再map中查找k对应的value //然后拿value,作为multimap 中的key, 在multimap中定位key是value的区间 //在区间内搜索哪个条目是记录map 中k 的记录 void erase(const key_type& k) { iterator itTarget = map.find(k); if (itTarget == map.end()) return; std::pair<rmap_iterator, rmap_iterator> itPair = rmap.equal_range(itTarget->second); for (rmap_iterator it = itPair.first; it != itPair.second; ++it) if (it->second == itTarget) { rmap.erase(it); map.erase(itTarget); return; } // Shouldn't ever get here assert(0); } //类似insert 的逻辑 void update(const_iterator itIn, const mapped_type& v) { // Using map::erase() with empty range instead of map::find() to get a non-const iterator, // since it is a constant time operation in C++11. For more details, see // https://stackoverflow.com/questions/765148/how-to-remove-constness-of-const-iterator iterator itTarget = map.erase(itIn, itIn); if (itTarget == map.end()) return; std::pair<rmap_iterator, rmap_iterator> itPair = rmap.equal_range(itTarget->second); for (rmap_iterator it = itPair.first; it != itPair.second; ++it) if (it->second == itTarget) { rmap.erase(it); itTarget->second = v; rmap.insert(make_pair(v, itTarget)); return; } // Shouldn't ever get here assert(0); } size_type max_size() const { return nMaxSize; } //overload max_size 成员, 调整max limit 数目 size_type max_size(size_type s) { assert(s > 0); while (map.size() > s) { map.erase(rmap.begin()->second); rmap.erase(rmap.begin()); } nMaxSize = s; return nMaxSize; } };
“bitcoin中的limitedmap有什么用”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注亿速云网站,小编将为大家输出更多高质量的实用文章!
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。