XTL  0.1
eXtended Template Library
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
hash_map.hpp
Go to the documentation of this file.
1 
6 #pragma once
8 
9 #include <cstdint>
10 #include <vector>
11 #include <atomic>
12 
13 #include <xtd/meta.hpp>
14 
15 namespace xtd{
16 
17  namespace concurrent {
25  template<typename _HashMapT>
26  class hash_map_iterator {
27  template<typename, typename, int> friend
28  class hash_map;
29 
30  static constexpr int key_nibbles = sizeof(typename _HashMapT::key_type) * 2;
31  const _HashMapT *_Map;
32  typename _HashMapT::value_type *_Current;
33  std::vector<int8_t> _Key;
34 
35  hash_map_iterator(const _HashMapT *pMap, typename _HashMapT::value_type *pCurrent,
36  const std::vector<int8_t> &oKey) : _Map(pMap), _Current(pCurrent), _Key(oKey) {}
37 
38  public:
39  using value_type = typename _HashMapT::value_type;
40 
41  hash_map_iterator(const hash_map_iterator &src) : _Map(src._Map), _Current(src._Current), _Key(src._Key) {}
42 
43  hash_map_iterator(hash_map_iterator &&src) : _Map(src._Map), _Current(src._Current), _Key(std::move(src._Key)) {}
44 
45  hash_map_iterator() : _Map(nullptr), _Current(nullptr), _Key(key_nibbles, -1) {}
46 
47  hash_map_iterator &operator=(const hash_map_iterator &src) {
48  if (this == &src) {
49  return *this;
50  }
51  _Map = src._Map;
52  _Current = src._Current;
53  _Key = src._Key;
54  }
55 
56  hash_map_iterator &operator==(hash_map_iterator &&src) {
57  if (this == &src) {
58  return *this;
59  }
60  _Map = src._Map;
61  _Current = src._Current;
62  _Key = std::move(src._Key);
63  }
64 
65  bool operator==(const hash_map_iterator &rhs) const {
66  return (_Current == rhs._Current);
67  }
68 
69  bool operator!=(const hash_map_iterator &rhs) const {
70  return (_Current != rhs._Current);
71  }
72 
73  value_type *get() { return _Current; }
74 
75  const value_type *get() const { return _Current; }
76 
77  value_type *operator->() { return _Current; }
78 
79  const value_type *operator->() const { return _Current; }
80 
81  value_type &operator*() {
82  XTD_ASSERT(_Current);
83  return *_Current;
84  }
85 
86  const value_type &operator*() const {
87  XTD_ASSERT(_Current);
88  return *_Current;
89  }
90 
91  hash_map_iterator &operator++() {
92  ++_Key.back();
93  _Current = _Map->_next(&_Key[0]);
94  return *this;
95  }
96 
97  hash_map_iterator operator++(int) {
98  hash_map_iterator oRet(*this);
99  ++*this;
100  return oRet;
101  }
102 
103  hash_map_iterator &operator--() {
104  --_Key.back();
105  _Current = _Map->_prev(&_Key[0]);
106  return *this;
107  }
108 
109  hash_map_iterator operator--(int) {
110  hash_map_iterator oRet(*this);
111  --*this;
112  return oRet;
113  }
114 
115  };
116 
117 
118 
125  template<typename _KeyT, typename _ValueT, int _NibblePos = sizeof(_KeyT) * 2>
126  class hash_map {
127  using _my_t = hash_map<_KeyT, _ValueT, _NibblePos>;
128  using child_bucket_type = hash_map<_KeyT, _ValueT, _NibblePos - 1>;
129  static constexpr int nibble_pos = _NibblePos;
130  template<typename> friend
131  class hash_map_iterator;
132 
133  template<typename, typename, int> friend
134  class hash_map;
135 
136  static constexpr int8_t nibble_count = 16;
137  std::atomic<child_bucket_type *> _Buckets[nibble_count];
138 
139  public:
140  using value_type = _ValueT;
141  using key_type = _KeyT;
142  using iterator_type = hash_map_iterator<_my_t>;
143 
144  hash_map() {
145  for (auto &oItem : _Buckets) {
146  oItem.store(nullptr);
147  }
148  }
149 
150  ~hash_map(){
151  for (int8_t i=0 ; i<nibble_count ; ++i){
152  auto pItem = _Buckets[i].load();
153  if (pItem){
154  delete pItem;
155  }
156  }
157  }
158 
159  hash_map(const hash_map &) = delete;
160 
161  hash_map &operator=(const hash_map &) = delete;
162 
168  bool insert(const key_type &Key, value_type &&Value) {
169  auto x = intrinsic_cast(Key);
170  int Index = (x & 0xf);
171  auto pChild = _Buckets[Index].load();
172  if (!pChild) {
173  pChild = new child_bucket_type;
174  child_bucket_type *pNullBucket = nullptr;
175  if (!_Buckets[Index].compare_exchange_strong(pNullBucket, pChild)) {
176  delete pChild;
177  }
178  }
179  x >>= 4;
180  return pChild->insert(intrinsic_cast(x), std::forward<value_type>(Value));
181  }
182 
187  bool exists(const key_type &Key) const {
188  auto x = intrinsic_cast(Key);
189  int Index = (x & 0xf);
190  auto pChild = _Buckets[Index].load();
191  x >>= 4;
192  return (pChild ? pChild->exists(intrinsic_cast(x)) : false);
193  }
194 
199  bool remove(const key_type &Key) {
200  auto x = intrinsic_cast(Key);
201  int Index = (x & 0xf);
202  auto pChild = _Buckets[Index].load();
203  if (pChild) {
204  x >>= 4;
205  return pChild->remove(intrinsic_cast(x));
206  }
207  return false;
208  }
209 
215  value_type &operator[](const key_type &Key) {
216  auto x = intrinsic_cast(Key);
217  int Index = (x & 0xf);
218  auto pChild = _Buckets[Index].load();
219  if (!pChild) {
220  insert(Key, value_type());
221  pChild = _Buckets[Index].load();
222  }
223  x >>= 4;
224  return pChild->operator[](intrinsic_cast(x));
225  }
226 
228  iterator_type begin() const {
229  std::vector<int8_t> oKey(iterator_type::key_nibbles, -1);
230  return iterator_type(this, _begin(&oKey[0]), oKey);
231  }
232 
234  iterator_type end() const {
235  return iterator_type(this, nullptr, std::vector<int8_t>(iterator_type::key_nibbles, -1));
236  }
237 
239  iterator_type back() const {
240  std::vector<int8_t> oKey(iterator_type::key_nibbles, -1);
241  return iterator_type(this, _back(&oKey[0]), oKey);
242  }
243 
244  private:
245 
246  value_type *_begin(int8_t *pKey) const {
247  child_bucket_type *pChildBucket;
248  for (*pKey = 0; *pKey < nibble_count; ++*pKey) {
249  value_type *pRet;
250  if ((pChildBucket = _Buckets[*pKey].load()) && (pRet = pChildBucket->_begin(1 + pKey))) {
251  return pRet;
252  }
253  }
254  return nullptr;
255  }
256 
257  value_type *_back(int8_t *pKey) const {
258  child_bucket_type *pChildBucket;
259  for (*pKey = nibble_count - 1; *pKey >= 0; --*pKey) {
260  value_type *pRet;
261  if ((pChildBucket = _Buckets[*pKey].load()) && (pRet = pChildBucket->_back(1 + pKey))) {
262  return pRet;
263  }
264  }
265  return nullptr;
266  }
267 
268  value_type *_next(int8_t *pKey) const {
269  child_bucket_type *pChildBucket;
270  if (*pKey < 0 || *pKey >= nibble_count) {
271  *pKey = 0;
272  }
273  for (; *pKey < nibble_count; ++*pKey) {
274  value_type *pRet;
275  if ((pChildBucket = _Buckets[*pKey].load()) && (pRet = pChildBucket->_next(1 + pKey))) {
276  return pRet;
277  }
278  }
279  *pKey = -1;
280  return nullptr;
281  }
282 
283  value_type *_prev(int8_t *pKey) const {
284  child_bucket_type *pChildBucket;
285  if (*pKey < 0 || *pKey >= nibble_count) {
286  *pKey = nibble_count - 1;
287  }
288  for (; *pKey >= 0; --*pKey) {
289  value_type *pRet;
290  if ((pChildBucket = _Buckets[*pKey].load()) && (pRet = pChildBucket->_prev(1 + pKey))) {
291  return pRet;
292  }
293  }
294  *pKey = -1;
295  return nullptr;
296  }
297 
298  };
299 
300 #if (!DOXY_INVOKED)
301 
302  template<typename _KeyT, typename _ValueT>
303  class hash_map<_KeyT, _ValueT, 1> {
304  template<typename, typename, int> friend
305  class hash_map;
306 
307  static constexpr int nibble_pos = 1;
308  static constexpr int8_t nibble_count = 16;
309  std::atomic<_ValueT *> _Values[nibble_count];
310  public:
311  using _my_t = hash_map<_KeyT, _ValueT, 1>;
312  using value_type = _ValueT;
313  using key_type = _KeyT;
314 
315 
316  hash_map() {
317  for (auto &oItem : _Values) {
318  oItem.store(nullptr);
319  }
320  }
321 
322  ~hash_map(){
323  for (int8_t i=0 ; i<nibble_count ; ++i){
324  auto pItem = _Values[i].load();
325  if (pItem){
326  delete pItem;
327  }
328  }
329  }
330 
331  bool insert(const key_type &Key, value_type &&Value) {
332  auto x = intrinsic_cast(Key);
333  int Index = (x & 0xf);
334  auto pValue = _Values[Index].load();
335  if (pValue) {
336  return false;
337  }
338  pValue = new value_type(std::forward<value_type>(Value));
339  value_type *pNullValue = nullptr;
340  if (!_Values[Index].compare_exchange_strong(pNullValue, pValue)) {
341  delete pValue;
342  return false;
343  }
344  return true;
345  }
346 
347  bool remove(const key_type &Key) {
348  auto x = intrinsic_cast(Key);
349  int Index = (x & 0xf);
350  auto pVal = _Values[Index].load();
351  value_type *pNullValue = nullptr;
352  if (pVal && _Values[Index].compare_exchange_strong(pVal, pNullValue)) {
353  delete pVal;
354  return true;
355  }
356  return false;
357  }
358 
359  bool exists(const key_type &Key) const {
360  auto x = intrinsic_cast(Key);
361  int Index = (x & 0xf);
362  auto pVal = _Values[Index].load();
363  return (pVal ? true : false);
364  }
365 
366  value_type &operator[](const key_type &Key) {
367  auto x = intrinsic_cast(Key);
368  int Index = (x & 0xf);
369  auto pRet = _Values[Index].load();
370  if (!pRet) {
371  pRet = new value_type;
372  value_type *pNull = nullptr;
373  if (!_Values[Index].compare_exchange_strong(pNull, pRet)) {
374  delete pRet;
375  pRet = _Values[Index].load();
376  }
377  }
378  XTD_ASSERT(pRet);
379  return *pRet;
380  }
381 
382  private:
383 
384  value_type *_begin(int8_t *pKey) const {
385  for (*pKey = 0; *pKey < nibble_count; ++*pKey) {
386  value_type *pRet;
387  if ((pRet = _Values[*pKey].load())) {
388  return pRet;
389  }
390  }
391  return nullptr;
392  }
393 
394  value_type *_back(int8_t *pKey) const {
395  for (*pKey = nibble_count - 1; *pKey >= 0; --*pKey) {
396  value_type *pRet;
397  if ((pRet = _Values[*pKey].load())) {
398  return pRet;
399  }
400  }
401  return nullptr;
402  }
403 
404  value_type *_next(int8_t *pKey) const {
405  if (*pKey < 0 || *pKey >= nibble_count) {
406  *pKey = 0;
407  }
408  for (; *pKey < nibble_count; ++*pKey) {
409  value_type *pRet;
410  if ((pRet = _Values[*pKey].load())) {
411  return pRet;
412  }
413  }
414  *pKey = -1;
415  return nullptr;
416  }
417 
418  value_type *_prev(int8_t *pKey) const {
419  if (*pKey < 0 || *pKey >= nibble_count) {
420  *pKey = nibble_count - 1;
421  }
422  for (; *pKey >= 0; --*pKey) {
423  value_type *pRet;
424  if ((pRet = _Values[*pKey].load())) {
425  return pRet;
426  }
427  }
428  *pKey = -1;
429  return nullptr;
430  }
431 
432  };
433  }
434 #endif
435 
437 
438 
439 
440 }
441 
template meta-programming utilities
shared declarations for the concurrent namespace
constexpr processor_intrinsic< _Ty >::type intrinsic_cast(_Ty src)
casts a pointer to the processor intrinsic storage type
Definition: meta.hpp:70