您好,登錄后才能下訂單哦!
本篇內容介紹了“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進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。