veriblock-pop-cpp
C++11 Libraries for leveraging VeriBlock Proof-Of-Proof blockchain technology.
value_sorted_map.hpp
1// Copyright (c) 2019-2022 Xenios SEZC
2// https://www.veriblock.org
3// Distributed under the MIT software license, see the accompanying
4// file LICENSE or http://www.opensource.org/licenses/mit-license.php.
5
6#ifndef ALTINTEGRATION_VALUE_SORTED_MAP___
7#define ALTINTEGRATION_VALUE_SORTED_MAP___
8
9#include <algorithm>
10#include <functional>
11#include <set>
12#include <unordered_map>
13
14#include "assert.hpp"
15
16namespace altintegration {
17
19template <typename K, typename V>
20class ValueSortedMap {
21 public:
22 using pair_t = std::pair<const K, V>;
23 using cmp_t = std::function<bool(const V&, const V&)>;
24 using set_t = typename std::multiset<V, cmp_t>;
25 using map_t = std::unordered_map<K, V>;
26
27 private:
28 set_t set_;
29 map_t map_;
30
31 public:
32 using iterator_t = typename map_t::iterator;
33 using const_iterator_t = typename map_t::const_iterator;
34
35 ~ValueSortedMap() = default;
36
37 explicit ValueSortedMap(cmp_t comparator) : set_(comparator) {}
38
39 iterator_t find(const K& key) { return map_.find(key); }
40 const_iterator_t find(const K& key) const { return map_.find(key); }
41
42 auto begin() -> iterator_t { return map_.begin(); }
43 auto end() -> iterator_t { return map_.end(); }
44 auto begin() const -> const_iterator_t { return map_.begin(); }
45 auto end() const -> const_iterator_t { return map_.end(); }
46
47 const set_t& getSortedValues() const { return set_; }
48
49 // implicit const map_t& cast operator
50 operator const map_t&() const { return map_; }
51
52 void erase(const K& key) {
53 auto map_it = map_.find(key);
54 if (map_it == map_.end()) {
55 return;
56 }
57
58 auto set_it = set_.find(map_it->second);
59 VBK_ASSERT(set_it != set_.end());
60 set_.erase(set_it);
61 map_.erase(map_it);
62
63 VBK_ASSERT_MSG(map_.size() == set_.size(),
64 "size of map and set are different map: %d, set: %d",
65 map_.size(),
66 set_.size());
67 }
68
69 iterator_t erase(const iterator_t& it) {
70 set_.erase(set_.find(it->second));
71 auto res = map_.erase(it);
72
73 VBK_ASSERT_MSG(map_.size() == set_.size(),
74 "size of map and set are different map: %d, set: %d",
75 map_.size(),
76 set_.size());
77
78 return res;
79 }
80
81 void insert(const K& key, const V& value) {
82 pair_t pair(key, value);
83
84 auto map_it = map_.find(key);
85 if (map_it != map_.end()) {
86 // key exists
87 auto set_it = set_.find(map_it->second);
88 VBK_ASSERT(set_it != set_.end());
89 set_.erase(set_it);
90
91 map_it->second = value;
92 set_.insert(map_it->second);
93 } else {
94 auto res = map_.insert(std::move(pair));
95 VBK_ASSERT(res.second);
96 auto it = res.first;
97 set_.insert(it->second);
98 }
99
100 VBK_ASSERT_MSG(map_.size() == set_.size(),
101 "size of map and set are different map: %d, set: %d",
102 map_.size(),
103 set_.size());
104 }
105
106 size_t size() const {
107 VBK_ASSERT_MSG(map_.size() == set_.size(),
108 "size of map and set are different map: %d, set: %d",
109 map_.size(),
110 set_.size());
111
112 return map_.size();
113 }
114
115 void clear() {
116 VBK_ASSERT_MSG(map_.size() == set_.size(),
117 "size of map and set are different map: %d, set: %d",
118 map_.size(),
119 set_.size());
120 set_.clear();
121 map_.clear();
122 }
123
124 bool empty() const {
125 VBK_ASSERT_MSG(map_.size() == set_.size(),
126 "size of map and set are different map: %d, set: %d",
127 map_.size(),
128 set_.size());
129
130 return map_.empty();
131 }
132};
133
134} // namespace altintegration
135
136#endif
Defines logging helpers.
Definition: block.hpp:14