// Copyright (c) 2014-2017 The Dash Core developers // Distributed under the MIT/X11 software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #ifndef CACHEMULTIMAP_H_ #define CACHEMULTIMAP_H_ #include #include #include #include #include "serialize.h" #include "cachemap.h" /** * Map like container that keeps the N most recently added items */ template class CacheMultiMap { public: typedef Size size_type; typedef CacheItem item_t; typedef std::list list_t; typedef typename list_t::iterator list_it; typedef typename list_t::const_iterator list_cit; typedef std::map it_map_t; typedef typename it_map_t::iterator it_map_it; typedef typename it_map_t::const_iterator it_map_cit; typedef std::map map_t; typedef typename map_t::iterator map_it; typedef typename map_t::const_iterator map_cit; private: size_type nMaxSize; size_type nCurrentSize; list_t listItems; map_t mapIndex; public: CacheMultiMap(size_type nMaxSizeIn = 0) : nMaxSize(nMaxSizeIn), nCurrentSize(0), listItems(), mapIndex() {} CacheMultiMap(const CacheMap& other) : nMaxSize(other.nMaxSize), nCurrentSize(other.nCurrentSize), listItems(other.listItems), mapIndex() { RebuildIndex(); } void Clear() { mapIndex.clear(); listItems.clear(); nCurrentSize = 0; } void SetMaxSize(size_type nMaxSizeIn) { nMaxSize = nMaxSizeIn; } size_type GetMaxSize() const { return nMaxSize; } size_type GetSize() const { return nCurrentSize; } bool Insert(const K& key, const V& value) { if(nCurrentSize == nMaxSize) { PruneLast(); } map_it mit = mapIndex.find(key); if(mit == mapIndex.end()) { mit = mapIndex.insert(std::pair(key, it_map_t())).first; } it_map_t& mapIt = mit->second; if(mapIt.count(value) > 0) { // Don't insert duplicates return false; } listItems.push_front(item_t(key, value)); list_it lit = listItems.begin(); mapIt[value] = lit; ++nCurrentSize; return true; } bool HasKey(const K& key) const { map_cit it = mapIndex.find(key); return (it != mapIndex.end()); } bool Get(const K& key, V& value) const { map_cit it = mapIndex.find(key); if(it == mapIndex.end()) { return false; } const it_map_t& mapIt = it->second; const item_t& item = *(mapIt.begin()->second); value = item.value; return true; } bool GetAll(const K& key, std::vector& vecValues) { map_cit mit = mapIndex.find(key); if(mit == mapIndex.end()) { return false; } const it_map_t& mapIt = mit->second; for(it_map_cit it = mapIt.begin(); it != mapIt.end(); ++it) { const item_t& item = *(it->second); vecValues.push_back(item.value); } return true; } void GetKeys(std::vector& vecKeys) { for(map_cit it = mapIndex.begin(); it != mapIndex.end(); ++it) { vecKeys.push_back(it->first); } } void Erase(const K& key) { map_it mit = mapIndex.find(key); if(mit == mapIndex.end()) { return; } it_map_t& mapIt = mit->second; for(it_map_it it = mapIt.begin(); it != mapIt.end(); ++it) { listItems.erase(it->second); --nCurrentSize; } mapIndex.erase(mit); } void Erase(const K& key, const V& value) { map_it mit = mapIndex.find(key); if(mit == mapIndex.end()) { return; } it_map_t& mapIt = mit->second; it_map_it it = mapIt.find(value); if(it == mapIt.end()) { return; } listItems.erase(it->second); --nCurrentSize; mapIt.erase(it); if(mapIt.size() < 1) { mapIndex.erase(mit); } } const list_t& GetItemList() const { return listItems; } CacheMap& operator=(const CacheMap& other) { nMaxSize = other.nMaxSize; nCurrentSize = other.nCurrentSize; listItems = other.listItems; RebuildIndex(); return *this; } ADD_SERIALIZE_METHODS; template inline void SerializationOp(Stream& s, Operation ser_action) { READWRITE(nMaxSize); READWRITE(nCurrentSize); READWRITE(listItems); if(ser_action.ForRead()) { RebuildIndex(); } } private: void PruneLast() { if(nCurrentSize < 1) { return; } list_it lit = listItems.end(); --lit; item_t& item = *lit; map_it mit = mapIndex.find(item.key); if(mit != mapIndex.end()) { it_map_t& mapIt = mit->second; mapIt.erase(item.value); if(mapIt.size() < 1) { mapIndex.erase(item.key); } } listItems.pop_back(); --nCurrentSize; } void RebuildIndex() { mapIndex.clear(); for(list_it lit = listItems.begin(); lit != listItems.end(); ++lit) { item_t& item = *lit; map_it mit = mapIndex.find(item.key); if(mit == mapIndex.end()) { mit = mapIndex.insert(std::pair(item.key, it_map_t())).first; } it_map_t& mapIt = mit->second; mapIt[item.value] = lit; } } }; #endif /* CACHEMULTIMAP_H_ */