博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
bitcoin 中的 limitedmap
阅读量:6253 次
发布时间:2019-06-22

本文共 3047 字,大约阅读时间需要 10 分钟。

hot3.png

bitcoin 里面的limitedmap 基本上是在std::map 上实现了优先队列的功能, 主要用在记录向对等节点索取inv 对应的时间记录。 以前不了解, stl 容器可以如此灵活使用, 把多个容器组合一起,完成更强大的功能。limitedmap 的主要技巧是,使用一个multimap(limitedmap 中多个不同的key可能映射到同一个value), 记录map里面value及其在map的位置(迭代器), bitcoin 里面有不少这种用法, 一个容器的元素是另一个容器的迭代器。

template 
class limitedmap{public: typedef K key_type; typedef V mapped_type; typedef std::pair
value_type; typedef typename std::map
::const_iterator const_iterator; typedef typename std::map
::size_type size_type;protected: std::map
map; typedef typename std::map
::iterator iterator; std::multimap
rmap; typedef typename std::multimap
::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
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
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
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; }};

转载于:https://my.oschina.net/evilunix/blog/2872882

你可能感兴趣的文章
[用UpdateLayeredWindow实现任意异形窗口]
查看>>
Android 源码编译make的错误处理
查看>>
【Gson】2.2.4 StackOverflowError 异常
查看>>
来玩Play框架01 简介
查看>>
ArcSDE for Oracle表空间管理——暂时(TEMP)表空间
查看>>
Android Bundle类
查看>>
[转]IC行业的牛人
查看>>
linux 16进制 产看文件
查看>>
javaScript事件(四)event的公共成员(属性和方法)
查看>>
Oracle之比较NVARCHAR2字符串
查看>>
linux系统常用命令
查看>>
用原始方法解析复杂字符串,json一定要用JsonMapper么?
查看>>
Linux ld命令
查看>>
在 Word 中的受支持的区域设置标识符的列表
查看>>
No package的问题解决
查看>>
【转】chrome浏览器的跨域设置——包括版本49前后两种设置
查看>>
母牛的故事
查看>>
Caffe + Ubuntu 14.04 64bit + CUDA 6.5 配置说明2
查看>>
javaScript基础练习题-下拉框制作
查看>>
基于 OAuth 安全协议的 Java 应用编程1
查看>>