google_hash 0.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (85) hide show
  1. data/README +21 -0
  2. data/Rakefile +11 -0
  3. data/VERSION +1 -0
  4. data/ext/extconf.rb +15 -0
  5. data/ext/go.cpp +109 -0
  6. data/ext/sparsehash-1.5.2/AUTHORS +2 -0
  7. data/ext/sparsehash-1.5.2/COPYING +28 -0
  8. data/ext/sparsehash-1.5.2/ChangeLog +167 -0
  9. data/ext/sparsehash-1.5.2/INSTALL +236 -0
  10. data/ext/sparsehash-1.5.2/Makefile.am +157 -0
  11. data/ext/sparsehash-1.5.2/Makefile.in +1019 -0
  12. data/ext/sparsehash-1.5.2/NEWS +0 -0
  13. data/ext/sparsehash-1.5.2/README +149 -0
  14. data/ext/sparsehash-1.5.2/README.windows +25 -0
  15. data/ext/sparsehash-1.5.2/TODO +28 -0
  16. data/ext/sparsehash-1.5.2/aclocal.m4 +868 -0
  17. data/ext/sparsehash-1.5.2/compile +99 -0
  18. data/ext/sparsehash-1.5.2/config.guess +1516 -0
  19. data/ext/sparsehash-1.5.2/config.sub +1626 -0
  20. data/ext/sparsehash-1.5.2/configure +8054 -0
  21. data/ext/sparsehash-1.5.2/configure.ac +74 -0
  22. data/ext/sparsehash-1.5.2/depcomp +530 -0
  23. data/ext/sparsehash-1.5.2/doc/dense_hash_map.html +1591 -0
  24. data/ext/sparsehash-1.5.2/doc/dense_hash_set.html +1445 -0
  25. data/ext/sparsehash-1.5.2/doc/designstyle.css +115 -0
  26. data/ext/sparsehash-1.5.2/doc/implementation.html +365 -0
  27. data/ext/sparsehash-1.5.2/doc/index.html +69 -0
  28. data/ext/sparsehash-1.5.2/doc/performance.html +96 -0
  29. data/ext/sparsehash-1.5.2/doc/sparse_hash_map.html +1527 -0
  30. data/ext/sparsehash-1.5.2/doc/sparse_hash_set.html +1376 -0
  31. data/ext/sparsehash-1.5.2/doc/sparsetable.html +1393 -0
  32. data/ext/sparsehash-1.5.2/experimental/Makefile +9 -0
  33. data/ext/sparsehash-1.5.2/experimental/README +14 -0
  34. data/ext/sparsehash-1.5.2/experimental/example.c +54 -0
  35. data/ext/sparsehash-1.5.2/experimental/libchash.c +1537 -0
  36. data/ext/sparsehash-1.5.2/experimental/libchash.h +252 -0
  37. data/ext/sparsehash-1.5.2/google-sparsehash.sln +47 -0
  38. data/ext/sparsehash-1.5.2/install-sh +323 -0
  39. data/ext/sparsehash-1.5.2/m4/acx_pthread.m4 +363 -0
  40. data/ext/sparsehash-1.5.2/m4/google_namespace.m4 +42 -0
  41. data/ext/sparsehash-1.5.2/m4/namespaces.m4 +15 -0
  42. data/ext/sparsehash-1.5.2/m4/stl_hash.m4 +70 -0
  43. data/ext/sparsehash-1.5.2/m4/stl_hash_fun.m4 +36 -0
  44. data/ext/sparsehash-1.5.2/m4/stl_namespace.m4 +25 -0
  45. data/ext/sparsehash-1.5.2/missing +360 -0
  46. data/ext/sparsehash-1.5.2/mkinstalldirs +158 -0
  47. data/ext/sparsehash-1.5.2/packages/deb.sh +74 -0
  48. data/ext/sparsehash-1.5.2/packages/deb/README +7 -0
  49. data/ext/sparsehash-1.5.2/packages/deb/changelog +107 -0
  50. data/ext/sparsehash-1.5.2/packages/deb/compat +1 -0
  51. data/ext/sparsehash-1.5.2/packages/deb/control +17 -0
  52. data/ext/sparsehash-1.5.2/packages/deb/copyright +35 -0
  53. data/ext/sparsehash-1.5.2/packages/deb/docs +16 -0
  54. data/ext/sparsehash-1.5.2/packages/deb/rules +117 -0
  55. data/ext/sparsehash-1.5.2/packages/deb/sparsehash.dirs +2 -0
  56. data/ext/sparsehash-1.5.2/packages/deb/sparsehash.install +2 -0
  57. data/ext/sparsehash-1.5.2/packages/rpm.sh +86 -0
  58. data/ext/sparsehash-1.5.2/packages/rpm/rpm.spec +61 -0
  59. data/ext/sparsehash-1.5.2/src/config.h.in +131 -0
  60. data/ext/sparsehash-1.5.2/src/config.h.include +23 -0
  61. data/ext/sparsehash-1.5.2/src/google/dense_hash_map +310 -0
  62. data/ext/sparsehash-1.5.2/src/google/dense_hash_set +287 -0
  63. data/ext/sparsehash-1.5.2/src/google/sparse_hash_map +294 -0
  64. data/ext/sparsehash-1.5.2/src/google/sparse_hash_set +275 -0
  65. data/ext/sparsehash-1.5.2/src/google/sparsehash/densehashtable.h +1062 -0
  66. data/ext/sparsehash-1.5.2/src/google/sparsehash/sparsehashtable.h +1015 -0
  67. data/ext/sparsehash-1.5.2/src/google/sparsetable +1468 -0
  68. data/ext/sparsehash-1.5.2/src/google/type_traits.h +250 -0
  69. data/ext/sparsehash-1.5.2/src/hashtable_unittest.cc +1375 -0
  70. data/ext/sparsehash-1.5.2/src/simple_test.cc +103 -0
  71. data/ext/sparsehash-1.5.2/src/sparsetable_unittest.cc +696 -0
  72. data/ext/sparsehash-1.5.2/src/time_hash_map.cc +488 -0
  73. data/ext/sparsehash-1.5.2/src/type_traits_unittest.cc +492 -0
  74. data/ext/sparsehash-1.5.2/src/windows/config.h +149 -0
  75. data/ext/sparsehash-1.5.2/src/windows/google/sparsehash/sparseconfig.h +32 -0
  76. data/ext/sparsehash-1.5.2/src/windows/port.cc +63 -0
  77. data/ext/sparsehash-1.5.2/src/windows/port.h +81 -0
  78. data/ext/sparsehash-1.5.2/src/words +8944 -0
  79. data/ext/sparsehash-1.5.2/vsprojects/hashtable_unittest/hashtable_unittest.vcproj +187 -0
  80. data/ext/sparsehash-1.5.2/vsprojects/sparsetable_unittest/sparsetable_unittest.vcproj +172 -0
  81. data/ext/sparsehash-1.5.2/vsprojects/time_hash_map/time_hash_map.vcproj +187 -0
  82. data/ext/sparsehash-1.5.2/vsprojects/type_traits_unittest/type_traits_unittest.vcproj +169 -0
  83. data/ext/test.rb +10 -0
  84. data/test/spec.go +70 -0
  85. metadata +147 -0
@@ -0,0 +1,1015 @@
1
+ // Copyright (c) 2005, Google Inc.
2
+ // All rights reserved.
3
+ //
4
+ // Redistribution and use in source and binary forms, with or without
5
+ // modification, are permitted provided that the following conditions are
6
+ // met:
7
+ //
8
+ // * Redistributions of source code must retain the above copyright
9
+ // notice, this list of conditions and the following disclaimer.
10
+ // * Redistributions in binary form must reproduce the above
11
+ // copyright notice, this list of conditions and the following disclaimer
12
+ // in the documentation and/or other materials provided with the
13
+ // distribution.
14
+ // * Neither the name of Google Inc. nor the names of its
15
+ // contributors may be used to endorse or promote products derived from
16
+ // this software without specific prior written permission.
17
+ //
18
+ // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19
+ // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20
+ // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21
+ // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22
+ // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23
+ // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24
+ // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25
+ // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26
+ // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27
+ // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28
+ // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
+
30
+ // ---
31
+ // Author: Craig Silverstein
32
+ //
33
+ // A sparse hashtable is a particular implementation of
34
+ // a hashtable: one that is meant to minimize memory use.
35
+ // It does this by using a *sparse table* (cf sparsetable.h),
36
+ // which uses between 1 and 2 bits to store empty buckets
37
+ // (we may need another bit for hashtables that support deletion).
38
+ //
39
+ // When empty buckets are so cheap, an appealing hashtable
40
+ // implementation is internal probing, in which the hashtable
41
+ // is a single table, and collisions are resolved by trying
42
+ // to insert again in another bucket. The most cache-efficient
43
+ // internal probing schemes are linear probing (which suffers,
44
+ // alas, from clumping) and quadratic probing, which is what
45
+ // we implement by default.
46
+ //
47
+ // Deleted buckets are a bit of a pain. We have to somehow mark
48
+ // deleted buckets (the probing must distinguish them from empty
49
+ // buckets). The most principled way is to have another bitmap,
50
+ // but that's annoying and takes up space. Instead we let the
51
+ // user specify an "impossible" key. We set deleted buckets
52
+ // to have the impossible key.
53
+ //
54
+ // Note it is possible to change the value of the delete key
55
+ // on the fly; you can even remove it, though after that point
56
+ // the hashtable is insert_only until you set it again.
57
+ //
58
+ // You probably shouldn't use this code directly. Use
59
+ // <google/sparse_hash_table> or <google/sparse_hash_set> instead.
60
+ //
61
+ // You can modify the following, below:
62
+ // HT_OCCUPANCY_FLT -- how full before we double size
63
+ // HT_EMPTY_FLT -- how empty before we halve size
64
+ // HT_MIN_BUCKETS -- smallest bucket size
65
+ // HT_DEFAULT_STARTING_BUCKETS -- default bucket size at construct-time
66
+ //
67
+ // You can also change enlarge_resize_percent (which defaults to
68
+ // HT_OCCUPANCY_FLT), and shrink_resize_percent (which defaults to
69
+ // HT_EMPTY_FLT) with set_resizing_parameters().
70
+ //
71
+ // How to decide what values to use?
72
+ // shrink_resize_percent's default of .4 * OCCUPANCY_FLT, is probably good.
73
+ // HT_MIN_BUCKETS is probably unnecessary since you can specify
74
+ // (indirectly) the starting number of buckets at construct-time.
75
+ // For enlarge_resize_percent, you can use this chart to try to trade-off
76
+ // expected lookup time to the space taken up. By default, this
77
+ // code uses quadratic probing, though you can change it to linear
78
+ // via _JUMP below if you really want to.
79
+ //
80
+ // From http://www.augustana.ca/~mohrj/courses/1999.fall/csc210/lecture_notes/hashing.html
81
+ // NUMBER OF PROBES / LOOKUP Successful Unsuccessful
82
+ // Quadratic collision resolution 1 - ln(1-L) - L/2 1/(1-L) - L - ln(1-L)
83
+ // Linear collision resolution [1+1/(1-L)]/2 [1+1/(1-L)2]/2
84
+ //
85
+ // -- enlarge_resize_percent -- 0.10 0.50 0.60 0.75 0.80 0.90 0.99
86
+ // QUADRATIC COLLISION RES.
87
+ // probes/successful lookup 1.05 1.44 1.62 2.01 2.21 2.85 5.11
88
+ // probes/unsuccessful lookup 1.11 2.19 2.82 4.64 5.81 11.4 103.6
89
+ // LINEAR COLLISION RES.
90
+ // probes/successful lookup 1.06 1.5 1.75 2.5 3.0 5.5 50.5
91
+ // probes/unsuccessful lookup 1.12 2.5 3.6 8.5 13.0 50.0 5000.0
92
+ //
93
+ // The value type is required to be copy constructible and default
94
+ // constructible, but it need not be (and commonly isn't) assignable.
95
+
96
+ #ifndef _SPARSEHASHTABLE_H_
97
+ #define _SPARSEHASHTABLE_H_
98
+
99
+ #ifndef SPARSEHASH_STAT_UPDATE
100
+ #define SPARSEHASH_STAT_UPDATE(x) ((void) 0)
101
+ #endif
102
+
103
+ // The probing method
104
+ // Linear probing
105
+ // #define JUMP_(key, num_probes) ( 1 )
106
+ // Quadratic-ish probing
107
+ #define JUMP_(key, num_probes) ( num_probes )
108
+
109
+
110
+ #include <google/sparsehash/sparseconfig.h>
111
+ #include <assert.h>
112
+ #include <algorithm> // For swap(), eg
113
+ #include <iterator> // for facts about iterator tags
114
+ #include <utility> // for pair<>
115
+ #include <google/sparsetable> // Since that's basically what we are
116
+
117
+ _START_GOOGLE_NAMESPACE_
118
+
119
+ using STL_NAMESPACE::pair;
120
+
121
+ // Hashtable class, used to implement the hashed associative containers
122
+ // hash_set and hash_map.
123
+ //
124
+ // Value: what is stored in the table (each bucket is a Value).
125
+ // Key: something in a 1-to-1 correspondence to a Value, that can be used
126
+ // to search for a Value in the table (find() takes a Key).
127
+ // HashFcn: Takes a Key and returns an integer, the more unique the better.
128
+ // ExtractKey: given a Value, returns the unique Key associated with it.
129
+ // SetKey: given a Value* and a Key, modifies the value such that
130
+ // ExtractKey(value) == key. We guarantee this is only called
131
+ // with key == deleted_key.
132
+ // EqualKey: Given two Keys, says whether they are the same (that is,
133
+ // if they are both associated with the same Value).
134
+ // Alloc: STL allocator to use to allocate memory. Currently ignored.
135
+
136
+ template <class Value, class Key, class HashFcn,
137
+ class ExtractKey, class SetKey, class EqualKey, class Alloc>
138
+ class sparse_hashtable;
139
+
140
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
141
+ struct sparse_hashtable_iterator;
142
+
143
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
144
+ struct sparse_hashtable_const_iterator;
145
+
146
+ // As far as iterating, we're basically just a sparsetable
147
+ // that skips over deleted elements.
148
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
149
+ struct sparse_hashtable_iterator {
150
+ public:
151
+ typedef sparse_hashtable_iterator<V,K,HF,ExK,SetK,EqK,A> iterator;
152
+ typedef sparse_hashtable_const_iterator<V,K,HF,ExK,SetK,EqK,A> const_iterator;
153
+ typedef typename sparsetable<V>::nonempty_iterator st_iterator;
154
+
155
+ typedef STL_NAMESPACE::forward_iterator_tag iterator_category;
156
+ typedef V value_type;
157
+ typedef ptrdiff_t difference_type;
158
+ typedef size_t size_type;
159
+ typedef V& reference; // Value
160
+ typedef V* pointer;
161
+
162
+ // "Real" constructor and default constructor
163
+ sparse_hashtable_iterator(const sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> *h,
164
+ st_iterator it, st_iterator it_end)
165
+ : ht(h), pos(it), end(it_end) { advance_past_deleted(); }
166
+ sparse_hashtable_iterator() { } // not ever used internally
167
+ // The default destructor is fine; we don't define one
168
+ // The default operator= is fine; we don't define one
169
+
170
+ // Happy dereferencer
171
+ reference operator*() const { return *pos; }
172
+ pointer operator->() const { return &(operator*()); }
173
+
174
+ // Arithmetic. The only hard part is making sure that
175
+ // we're not on a marked-deleted array element
176
+ void advance_past_deleted() {
177
+ while ( pos != end && ht->test_deleted(*this) )
178
+ ++pos;
179
+ }
180
+ iterator& operator++() {
181
+ assert(pos != end); ++pos; advance_past_deleted(); return *this;
182
+ }
183
+ iterator operator++(int) { iterator tmp(*this); ++*this; return tmp; }
184
+
185
+ // Comparison.
186
+ bool operator==(const iterator& it) const { return pos == it.pos; }
187
+ bool operator!=(const iterator& it) const { return pos != it.pos; }
188
+
189
+
190
+ // The actual data
191
+ const sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> *ht;
192
+ st_iterator pos, end;
193
+ };
194
+
195
+ // Now do it all again, but with const-ness!
196
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
197
+ struct sparse_hashtable_const_iterator {
198
+ public:
199
+ typedef sparse_hashtable_iterator<V,K,HF,ExK,SetK,EqK,A> iterator;
200
+ typedef sparse_hashtable_const_iterator<V,K,HF,ExK,SetK,EqK,A> const_iterator;
201
+ typedef typename sparsetable<V>::const_nonempty_iterator st_iterator;
202
+
203
+ typedef STL_NAMESPACE::forward_iterator_tag iterator_category;
204
+ typedef V value_type;
205
+ typedef ptrdiff_t difference_type;
206
+ typedef size_t size_type;
207
+ typedef const V& reference; // Value
208
+ typedef const V* pointer;
209
+
210
+ // "Real" constructor and default constructor
211
+ sparse_hashtable_const_iterator(const sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> *h,
212
+ st_iterator it, st_iterator it_end)
213
+ : ht(h), pos(it), end(it_end) { advance_past_deleted(); }
214
+ // This lets us convert regular iterators to const iterators
215
+ sparse_hashtable_const_iterator() { } // never used internally
216
+ sparse_hashtable_const_iterator(const iterator &it)
217
+ : ht(it.ht), pos(it.pos), end(it.end) { }
218
+ // The default destructor is fine; we don't define one
219
+ // The default operator= is fine; we don't define one
220
+
221
+ // Happy dereferencer
222
+ reference operator*() const { return *pos; }
223
+ pointer operator->() const { return &(operator*()); }
224
+
225
+ // Arithmetic. The only hard part is making sure that
226
+ // we're not on a marked-deleted array element
227
+ void advance_past_deleted() {
228
+ while ( pos != end && ht->test_deleted(*this) )
229
+ ++pos;
230
+ }
231
+ const_iterator& operator++() {
232
+ assert(pos != end); ++pos; advance_past_deleted(); return *this;
233
+ }
234
+ const_iterator operator++(int) { const_iterator tmp(*this); ++*this; return tmp; }
235
+
236
+ // Comparison.
237
+ bool operator==(const const_iterator& it) const { return pos == it.pos; }
238
+ bool operator!=(const const_iterator& it) const { return pos != it.pos; }
239
+
240
+
241
+ // The actual data
242
+ const sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> *ht;
243
+ st_iterator pos, end;
244
+ };
245
+
246
+ // And once again, but this time freeing up memory as we iterate
247
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
248
+ struct sparse_hashtable_destructive_iterator {
249
+ public:
250
+ typedef sparse_hashtable_destructive_iterator<V,K,HF,ExK,SetK,EqK,A> iterator;
251
+ typedef typename sparsetable<V>::destructive_iterator st_iterator;
252
+
253
+ typedef STL_NAMESPACE::forward_iterator_tag iterator_category;
254
+ typedef V value_type;
255
+ typedef ptrdiff_t difference_type;
256
+ typedef size_t size_type;
257
+ typedef V& reference; // Value
258
+ typedef V* pointer;
259
+
260
+ // "Real" constructor and default constructor
261
+ sparse_hashtable_destructive_iterator(const
262
+ sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> *h,
263
+ st_iterator it, st_iterator it_end)
264
+ : ht(h), pos(it), end(it_end) { advance_past_deleted(); }
265
+ sparse_hashtable_destructive_iterator() { } // never used internally
266
+ // The default destructor is fine; we don't define one
267
+ // The default operator= is fine; we don't define one
268
+
269
+ // Happy dereferencer
270
+ reference operator*() const { return *pos; }
271
+ pointer operator->() const { return &(operator*()); }
272
+
273
+ // Arithmetic. The only hard part is making sure that
274
+ // we're not on a marked-deleted array element
275
+ void advance_past_deleted() {
276
+ while ( pos != end && ht->test_deleted(*this) )
277
+ ++pos;
278
+ }
279
+ iterator& operator++() {
280
+ assert(pos != end); ++pos; advance_past_deleted(); return *this;
281
+ }
282
+ iterator operator++(int) { iterator tmp(*this); ++*this; return tmp; }
283
+
284
+ // Comparison.
285
+ bool operator==(const iterator& it) const { return pos == it.pos; }
286
+ bool operator!=(const iterator& it) const { return pos != it.pos; }
287
+
288
+
289
+ // The actual data
290
+ const sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> *ht;
291
+ st_iterator pos, end;
292
+ };
293
+
294
+
295
+ template <class Value, class Key, class HashFcn,
296
+ class ExtractKey, class SetKey, class EqualKey, class Alloc>
297
+ class sparse_hashtable {
298
+ public:
299
+ typedef Key key_type;
300
+ typedef Value value_type;
301
+ typedef HashFcn hasher;
302
+ typedef EqualKey key_equal;
303
+
304
+ typedef size_t size_type;
305
+ typedef ptrdiff_t difference_type;
306
+ typedef value_type* pointer;
307
+ typedef const value_type* const_pointer;
308
+ typedef value_type& reference;
309
+ typedef const value_type& const_reference;
310
+ typedef sparse_hashtable_iterator<Value, Key, HashFcn, ExtractKey,
311
+ SetKey, EqualKey, Alloc>
312
+ iterator;
313
+
314
+ typedef sparse_hashtable_const_iterator<Value, Key, HashFcn, ExtractKey,
315
+ SetKey, EqualKey, Alloc>
316
+ const_iterator;
317
+
318
+ typedef sparse_hashtable_destructive_iterator<Value, Key, HashFcn, ExtractKey,
319
+ SetKey, EqualKey, Alloc>
320
+ destructive_iterator;
321
+
322
+ // These come from tr1. For us they're the same as regular iterators.
323
+ typedef iterator local_iterator;
324
+ typedef const_iterator const_local_iterator;
325
+
326
+ // How full we let the table get before we resize, by default.
327
+ // Knuth says .8 is good -- higher causes us to probe too much,
328
+ // though it saves memory.
329
+ static const float HT_OCCUPANCY_FLT; // = 0.8f;
330
+
331
+ // How empty we let the table get before we resize lower, by default.
332
+ // It should be less than OCCUPANCY_FLT / 2 or we thrash resizing
333
+ static const float HT_EMPTY_FLT; // = 0.4 * HT_OCCUPANCY_FLT;
334
+
335
+ // Minimum size we're willing to let hashtables be.
336
+ // Must be a power of two, and at least 4.
337
+ // Note, however, that for a given hashtable, the minimum size is
338
+ // determined by the first constructor arg, and may be >HT_MIN_BUCKETS.
339
+ static const size_t HT_MIN_BUCKETS = 4;
340
+
341
+ // By default, if you don't specify a hashtable size at
342
+ // construction-time, we use this size. Must be a power of two, and
343
+ // at least HT_MIN_BUCKETS.
344
+ static const size_t HT_DEFAULT_STARTING_BUCKETS = 32;
345
+
346
+ // ITERATOR FUNCTIONS
347
+ iterator begin() { return iterator(this, table.nonempty_begin(),
348
+ table.nonempty_end()); }
349
+ iterator end() { return iterator(this, table.nonempty_end(),
350
+ table.nonempty_end()); }
351
+ const_iterator begin() const { return const_iterator(this,
352
+ table.nonempty_begin(),
353
+ table.nonempty_end()); }
354
+ const_iterator end() const { return const_iterator(this,
355
+ table.nonempty_end(),
356
+ table.nonempty_end()); }
357
+
358
+ // These come from tr1 unordered_map. They iterate over 'bucket' n.
359
+ // For sparsehashtable, we could consider each 'group' to be a bucket,
360
+ // I guess, but I don't really see the point. We'll just consider
361
+ // bucket n to be the n-th element of the sparsetable, if it's occupied,
362
+ // or some empty element, otherwise.
363
+ local_iterator begin(size_type i) {
364
+ if (table.test(i))
365
+ return local_iterator(this, table.get_iter(i), table.nonempty_end());
366
+ else
367
+ return local_iterator(this, table.nonempty_end(), table.nonempty_end());
368
+ }
369
+ local_iterator end(size_type i) {
370
+ local_iterator it = begin(i);
371
+ if (table.test(i) && !test_deleted(i))
372
+ ++it;
373
+ return it;
374
+ }
375
+ const_local_iterator begin(size_type i) const {
376
+ if (table.test(i))
377
+ return const_local_iterator(this, table.get_iter(i),
378
+ table.nonempty_end());
379
+ else
380
+ return const_local_iterator(this, table.nonempty_end(),
381
+ table.nonempty_end());
382
+ }
383
+ const_local_iterator end(size_type i) const {
384
+ const_local_iterator it = begin(i);
385
+ if (table.test(i) && !test_deleted(i))
386
+ ++it;
387
+ return it;
388
+ }
389
+
390
+ // This is used when resizing
391
+ destructive_iterator destructive_begin() {
392
+ return destructive_iterator(this, table.destructive_begin(),
393
+ table.destructive_end());
394
+ }
395
+ destructive_iterator destructive_end() {
396
+ return destructive_iterator(this, table.destructive_end(),
397
+ table.destructive_end());
398
+ }
399
+
400
+
401
+ // ACCESSOR FUNCTIONS for the things we templatize on, basically
402
+ hasher hash_funct() const { return hash; }
403
+ key_equal key_eq() const { return equals; }
404
+
405
+ private:
406
+ // We need to copy values when we set the special marker for deleted
407
+ // elements, but, annoyingly, we can't just use the copy assignment
408
+ // operator because value_type might not be assignable (it's often
409
+ // pair<const X, Y>). We use explicit destructor invocation and
410
+ // placement new to get around this. Arg.
411
+ void set_value(value_type* dst, const value_type src) {
412
+ dst->~value_type(); // delete the old value, if any
413
+ new(dst) value_type(src);
414
+ }
415
+
416
+ // This is used as a tag for the copy constructor, saying to destroy its
417
+ // arg We have two ways of destructively copying: with potentially growing
418
+ // the hashtable as we copy, and without. To make sure the outside world
419
+ // can't do a destructive copy, we make the typename private.
420
+ enum MoveDontCopyT {MoveDontCopy, MoveDontGrow};
421
+
422
+
423
+ // DELETE HELPER FUNCTIONS
424
+ // This lets the user describe a key that will indicate deleted
425
+ // table entries. This key should be an "impossible" entry --
426
+ // if you try to insert it for real, you won't be able to retrieve it!
427
+ // (NB: while you pass in an entire value, only the key part is looked
428
+ // at. This is just because I don't know how to assign just a key.)
429
+ private:
430
+ void squash_deleted() { // gets rid of any deleted entries we have
431
+ if ( num_deleted ) { // get rid of deleted before writing
432
+ sparse_hashtable tmp(MoveDontGrow, *this);
433
+ swap(tmp); // now we are tmp
434
+ }
435
+ assert(num_deleted == 0);
436
+ }
437
+
438
+ public:
439
+ void set_deleted_key(const key_type &key) {
440
+ // It's only safe to change what "deleted" means if we purge deleted guys
441
+ squash_deleted();
442
+ use_deleted = true;
443
+ delkey = key;
444
+ }
445
+ void clear_deleted_key() {
446
+ squash_deleted();
447
+ use_deleted = false;
448
+ }
449
+
450
+ // These are public so the iterators can use them
451
+ // True if the item at position bucknum is "deleted" marker
452
+ bool test_deleted(size_type bucknum) const {
453
+ // The num_deleted test is crucial for read(): after read(), the ht values
454
+ // are garbage, and we don't want to think some of them are deleted.
455
+ return (use_deleted && num_deleted > 0 && table.test(bucknum) &&
456
+ equals(delkey, get_key(table.unsafe_get(bucknum))));
457
+ }
458
+ bool test_deleted(const iterator &it) const {
459
+ return (use_deleted && num_deleted > 0 &&
460
+ equals(delkey, get_key(*it)));
461
+ }
462
+ bool test_deleted(const const_iterator &it) const {
463
+ return (use_deleted && num_deleted > 0 &&
464
+ equals(delkey, get_key(*it)));
465
+ }
466
+ bool test_deleted(const destructive_iterator &it) const {
467
+ return (use_deleted && num_deleted > 0 &&
468
+ equals(delkey, get_key(*it)));
469
+ }
470
+ // Set it so test_deleted is true. true if object didn't used to be deleted
471
+ // See below (at erase()) to explain why we allow const_iterators
472
+ bool set_deleted(const_iterator &it) {
473
+ assert(use_deleted); // bad if set_deleted_key() wasn't called
474
+ bool retval = !test_deleted(it);
475
+ // &* converts from iterator to value-type
476
+ set_key(const_cast<value_type*>(&(*it)), delkey);
477
+ return retval;
478
+ }
479
+ // Set it so test_deleted is false. true if object used to be deleted
480
+ bool clear_deleted(const_iterator &it) {
481
+ assert(use_deleted); // bad if set_deleted_key() wasn't called
482
+ // happens automatically when we assign something else in its place
483
+ return test_deleted(it);
484
+ }
485
+
486
+
487
+ // FUNCTIONS CONCERNING SIZE
488
+ size_type size() const { return table.num_nonempty() - num_deleted; }
489
+ // Buckets are always a power of 2
490
+ size_type max_size() const { return (size_type(-1) >> 1U) + 1; }
491
+ bool empty() const { return size() == 0; }
492
+ size_type bucket_count() const { return table.size(); }
493
+ size_type max_bucket_count() const { return max_size(); }
494
+ // These are tr1 methods. Their idea of 'bucket' doesn't map well to
495
+ // what we do. We just say every bucket has 0 or 1 items in it.
496
+ size_type bucket_size(size_type i) const {
497
+ return begin(i) == end(i) ? 0 : 1;
498
+ }
499
+
500
+
501
+ private:
502
+ // Because of the above, size_type(-1) is never legal; use it for errors
503
+ static const size_type ILLEGAL_BUCKET = size_type(-1);
504
+
505
+ private:
506
+ // This is the smallest size a hashtable can be without being too crowded
507
+ // If you like, you can give a min #buckets as well as a min #elts
508
+ size_type min_size(size_type num_elts, size_type min_buckets_wanted) {
509
+ size_type sz = HT_MIN_BUCKETS;
510
+ while ( sz < min_buckets_wanted || num_elts >= sz * enlarge_resize_percent )
511
+ sz *= 2;
512
+ return sz;
513
+ }
514
+
515
+ // Used after a string of deletes
516
+ void maybe_shrink() {
517
+ assert(table.num_nonempty() >= num_deleted);
518
+ assert((bucket_count() & (bucket_count()-1)) == 0); // is a power of two
519
+ assert(bucket_count() >= HT_MIN_BUCKETS);
520
+
521
+ // If you construct a hashtable with < HT_DEFAULT_STARTING_BUCKETS,
522
+ // we'll never shrink until you get relatively big, and we'll never
523
+ // shrink below HT_DEFAULT_STARTING_BUCKETS. Otherwise, something
524
+ // like "dense_hash_set<int> x; x.insert(4); x.erase(4);" will
525
+ // shrink us down to HT_MIN_BUCKETS buckets, which is too small.
526
+ if (shrink_threshold > 0
527
+ && (table.num_nonempty()-num_deleted) < shrink_threshold &&
528
+ bucket_count() > HT_DEFAULT_STARTING_BUCKETS ) {
529
+ size_type sz = bucket_count() / 2; // find how much we should shrink
530
+ while ( sz > HT_DEFAULT_STARTING_BUCKETS &&
531
+ (table.num_nonempty() - num_deleted) <= sz *
532
+ shrink_resize_percent )
533
+ sz /= 2; // stay a power of 2
534
+ sparse_hashtable tmp(MoveDontCopy, *this, sz);
535
+ swap(tmp); // now we are tmp
536
+ }
537
+ consider_shrink = false; // because we just considered it
538
+ }
539
+
540
+ // We'll let you resize a hashtable -- though this makes us copy all!
541
+ // When you resize, you say, "make it big enough for this many more elements"
542
+ void resize_delta(size_type delta) {
543
+ if ( consider_shrink ) // see if lots of deletes happened
544
+ maybe_shrink();
545
+ if ( bucket_count() >= HT_MIN_BUCKETS &&
546
+ (table.num_nonempty() + delta) <= enlarge_threshold )
547
+ return; // we're ok as we are
548
+
549
+ // Sometimes, we need to resize just to get rid of all the
550
+ // "deleted" buckets that are clogging up the hashtable. So when
551
+ // deciding whether to resize, count the deleted buckets (which
552
+ // are currently taking up room). But later, when we decide what
553
+ // size to resize to, *don't* count deleted buckets, since they
554
+ // get discarded during the resize.
555
+ const size_type needed_size = min_size(table.num_nonempty() + delta, 0);
556
+ if ( needed_size > bucket_count() ) { // we don't have enough buckets
557
+ const size_type resize_to = min_size(table.num_nonempty() - num_deleted
558
+ + delta, 0);
559
+ sparse_hashtable tmp(MoveDontCopy, *this, resize_to);
560
+ swap(tmp); // now we are tmp
561
+ }
562
+ }
563
+
564
+ // Used to actually do the rehashing when we grow/shrink a hashtable
565
+ void copy_from(const sparse_hashtable &ht, size_type min_buckets_wanted) {
566
+ clear(); // clear table, set num_deleted to 0
567
+
568
+ // If we need to change the size of our table, do it now
569
+ const size_type resize_to = min_size(ht.size(), min_buckets_wanted);
570
+ if ( resize_to > bucket_count() ) { // we don't have enough buckets
571
+ table.resize(resize_to); // sets the number of buckets
572
+ reset_thresholds();
573
+ }
574
+
575
+ // We use a normal iterator to get non-deleted bcks from ht
576
+ // We could use insert() here, but since we know there are
577
+ // no duplicates and no deleted items, we can be more efficient
578
+ assert( (bucket_count() & (bucket_count()-1)) == 0); // a power of two
579
+ for ( const_iterator it = ht.begin(); it != ht.end(); ++it ) {
580
+ size_type num_probes = 0; // how many times we've probed
581
+ size_type bucknum;
582
+ const size_type bucket_count_minus_one = bucket_count() - 1;
583
+ for (bucknum = hash(get_key(*it)) & bucket_count_minus_one;
584
+ table.test(bucknum); // not empty
585
+ bucknum = (bucknum + JUMP_(key, num_probes)) & bucket_count_minus_one) {
586
+ ++num_probes;
587
+ assert(num_probes < bucket_count()); // or else the hashtable is full
588
+ }
589
+ table.set(bucknum, *it); // copies the value to here
590
+ }
591
+ }
592
+
593
+ // Implementation is like copy_from, but it destroys the table of the
594
+ // "from" guy by freeing sparsetable memory as we iterate. This is
595
+ // useful in resizing, since we're throwing away the "from" guy anyway.
596
+ void move_from(MoveDontCopyT mover, sparse_hashtable &ht,
597
+ size_type min_buckets_wanted) {
598
+ clear(); // clear table, set num_deleted to 0
599
+
600
+ // If we need to change the size of our table, do it now
601
+ size_t resize_to;
602
+ if ( mover == MoveDontGrow )
603
+ resize_to = ht.bucket_count(); // keep same size as old ht
604
+ else // MoveDontCopy
605
+ resize_to = min_size(ht.size(), min_buckets_wanted);
606
+ if ( resize_to > bucket_count() ) { // we don't have enough buckets
607
+ table.resize(resize_to); // sets the number of buckets
608
+ reset_thresholds();
609
+ }
610
+
611
+ // We use a normal iterator to get non-deleted bcks from ht
612
+ // We could use insert() here, but since we know there are
613
+ // no duplicates and no deleted items, we can be more efficient
614
+ assert( (bucket_count() & (bucket_count()-1)) == 0); // a power of two
615
+ // THIS IS THE MAJOR LINE THAT DIFFERS FROM COPY_FROM():
616
+ for ( destructive_iterator it = ht.destructive_begin();
617
+ it != ht.destructive_end(); ++it ) {
618
+ size_type num_probes = 0; // how many times we've probed
619
+ size_type bucknum;
620
+ for ( bucknum = hash(get_key(*it)) & (bucket_count()-1); // h % buck_cnt
621
+ table.test(bucknum); // not empty
622
+ bucknum = (bucknum + JUMP_(key, num_probes)) & (bucket_count()-1) ) {
623
+ ++num_probes;
624
+ assert(num_probes < bucket_count()); // or else the hashtable is full
625
+ }
626
+ table.set(bucknum, *it); // copies the value to here
627
+ }
628
+ }
629
+
630
+
631
+ // Required by the spec for hashed associative container
632
+ public:
633
+ // Though the docs say this should be num_buckets, I think it's much
634
+ // more useful as num_elements. As a special feature, calling with
635
+ // req_elements==0 will cause us to shrink if we can, saving space.
636
+ void resize(size_type req_elements) { // resize to this or larger
637
+ if ( consider_shrink || req_elements == 0 )
638
+ maybe_shrink();
639
+ if ( req_elements > table.num_nonempty() ) // we only grow
640
+ resize_delta(req_elements - table.num_nonempty());
641
+ }
642
+
643
+ // Get and change the value of shrink_resize_percent and
644
+ // enlarge_resize_percent. The description at the beginning of this
645
+ // file explains how to choose the values. Setting the shrink
646
+ // parameter to 0.0 ensures that the table never shrinks.
647
+ void get_resizing_parameters(float* shrink, float* grow) const {
648
+ *shrink = shrink_resize_percent;
649
+ *grow = enlarge_resize_percent;
650
+ }
651
+ void set_resizing_parameters(float shrink, float grow) {
652
+ assert(shrink >= 0.0);
653
+ assert(grow <= 1.0);
654
+ if (shrink > grow/2.0f)
655
+ shrink = grow / 2.0f; // otherwise we thrash hashtable size
656
+ shrink_resize_percent = shrink;
657
+ enlarge_resize_percent = grow;
658
+ reset_thresholds();
659
+ }
660
+
661
+ // CONSTRUCTORS -- as required by the specs, we take a size,
662
+ // but also let you specify a hashfunction, key comparator,
663
+ // and key extractor. We also define a copy constructor and =.
664
+ // DESTRUCTOR -- the default is fine, surprisingly.
665
+ explicit sparse_hashtable(size_type expected_max_items_in_table = 0,
666
+ const HashFcn& hf = HashFcn(),
667
+ const EqualKey& eql = EqualKey(),
668
+ const SetKey& set = SetKey(),
669
+ const ExtractKey& ext = ExtractKey())
670
+ : hash(hf), equals(eql), get_key(ext), set_key(set), num_deleted(0),
671
+ use_deleted(false), delkey(), enlarge_resize_percent(HT_OCCUPANCY_FLT),
672
+ shrink_resize_percent(HT_EMPTY_FLT),
673
+ table(expected_max_items_in_table == 0
674
+ ? HT_DEFAULT_STARTING_BUCKETS
675
+ : min_size(expected_max_items_in_table, 0)) {
676
+ reset_thresholds();
677
+ }
678
+
679
+ // As a convenience for resize(), we allow an optional second argument
680
+ // which lets you make this new hashtable a different size than ht.
681
+ // We also provide a mechanism of saying you want to "move" the ht argument
682
+ // into us instead of copying.
683
+ sparse_hashtable(const sparse_hashtable& ht,
684
+ size_type min_buckets_wanted = HT_DEFAULT_STARTING_BUCKETS)
685
+ : hash(ht.hash), equals(ht.equals),
686
+ get_key(ht.get_key), set_key(ht.set_key), num_deleted(0),
687
+ use_deleted(ht.use_deleted), delkey(ht.delkey),
688
+ enlarge_resize_percent(ht.enlarge_resize_percent),
689
+ shrink_resize_percent(ht.shrink_resize_percent),
690
+ table() {
691
+ reset_thresholds();
692
+ copy_from(ht, min_buckets_wanted); // copy_from() ignores deleted entries
693
+ }
694
+ sparse_hashtable(MoveDontCopyT mover, sparse_hashtable& ht,
695
+ size_type min_buckets_wanted = HT_DEFAULT_STARTING_BUCKETS)
696
+ : hash(ht.hash), equals(ht.equals), get_key(ht.get_key),
697
+ num_deleted(0), use_deleted(ht.use_deleted), delkey(ht.delkey),
698
+ enlarge_resize_percent(ht.enlarge_resize_percent),
699
+ shrink_resize_percent(ht.shrink_resize_percent),
700
+ table() {
701
+ reset_thresholds();
702
+ move_from(mover, ht, min_buckets_wanted); // ignores deleted entries
703
+ }
704
+
705
+ sparse_hashtable& operator= (const sparse_hashtable& ht) {
706
+ if (&ht == this) return *this; // don't copy onto ourselves
707
+ clear();
708
+ hash = ht.hash;
709
+ equals = ht.equals;
710
+ get_key = ht.get_key;
711
+ set_key = ht.set_key;
712
+ use_deleted = ht.use_deleted;
713
+ delkey = ht.delkey;
714
+ copy_from(ht, HT_MIN_BUCKETS); // sets num_deleted to 0 too
715
+ return *this;
716
+ }
717
+
718
+ // Many STL algorithms use swap instead of copy constructors
719
+ void swap(sparse_hashtable& ht) {
720
+ STL_NAMESPACE::swap(hash, ht.hash);
721
+ STL_NAMESPACE::swap(equals, ht.equals);
722
+ STL_NAMESPACE::swap(get_key, ht.get_key);
723
+ STL_NAMESPACE::swap(set_key, ht.set_key);
724
+ STL_NAMESPACE::swap(num_deleted, ht.num_deleted);
725
+ STL_NAMESPACE::swap(use_deleted, ht.use_deleted);
726
+ STL_NAMESPACE::swap(enlarge_resize_percent, ht.enlarge_resize_percent);
727
+ STL_NAMESPACE::swap(shrink_resize_percent, ht.shrink_resize_percent);
728
+ STL_NAMESPACE::swap(delkey, ht.delkey);
729
+ table.swap(ht.table);
730
+ reset_thresholds();
731
+ ht.reset_thresholds();
732
+ }
733
+
734
+ // It's always nice to be able to clear a table without deallocating it
735
+ void clear() {
736
+ table.clear();
737
+ reset_thresholds();
738
+ num_deleted = 0;
739
+ }
740
+
741
+
742
+ // LOOKUP ROUTINES
743
+ private:
744
+ // Returns a pair of positions: 1st where the object is, 2nd where
745
+ // it would go if you wanted to insert it. 1st is ILLEGAL_BUCKET
746
+ // if object is not found; 2nd is ILLEGAL_BUCKET if it is.
747
+ // Note: because of deletions where-to-insert is not trivial: it's the
748
+ // first deleted bucket we see, as long as we don't find the key later
749
+ pair<size_type, size_type> find_position(const key_type &key) const {
750
+ size_type num_probes = 0; // how many times we've probed
751
+ const size_type bucket_count_minus_one = bucket_count() - 1;
752
+ size_type bucknum = hash(key) & bucket_count_minus_one;
753
+ size_type insert_pos = ILLEGAL_BUCKET; // where we would insert
754
+ SPARSEHASH_STAT_UPDATE(total_lookups += 1);
755
+ while ( 1 ) { // probe until something happens
756
+ if ( !table.test(bucknum) ) { // bucket is empty
757
+ SPARSEHASH_STAT_UPDATE(total_probes += num_probes);
758
+ if ( insert_pos == ILLEGAL_BUCKET ) // found no prior place to insert
759
+ return pair<size_type,size_type>(ILLEGAL_BUCKET, bucknum);
760
+ else
761
+ return pair<size_type,size_type>(ILLEGAL_BUCKET, insert_pos);
762
+
763
+ } else if ( test_deleted(bucknum) ) {// keep searching, but mark to insert
764
+ if ( insert_pos == ILLEGAL_BUCKET )
765
+ insert_pos = bucknum;
766
+
767
+ } else if ( equals(key, get_key(table.unsafe_get(bucknum))) ) {
768
+ SPARSEHASH_STAT_UPDATE(total_probes += num_probes);
769
+ return pair<size_type,size_type>(bucknum, ILLEGAL_BUCKET);
770
+ }
771
+ ++num_probes; // we're doing another probe
772
+ bucknum = (bucknum + JUMP_(key, num_probes)) & bucket_count_minus_one;
773
+ assert(num_probes < bucket_count()); // don't probe too many times!
774
+ }
775
+ }
776
+
777
+ public:
778
+ iterator find(const key_type& key) {
779
+ if ( size() == 0 ) return end();
780
+ pair<size_type, size_type> pos = find_position(key);
781
+ if ( pos.first == ILLEGAL_BUCKET ) // alas, not there
782
+ return end();
783
+ else
784
+ return iterator(this, table.get_iter(pos.first), table.nonempty_end());
785
+ }
786
+
787
+ const_iterator find(const key_type& key) const {
788
+ if ( size() == 0 ) return end();
789
+ pair<size_type, size_type> pos = find_position(key);
790
+ if ( pos.first == ILLEGAL_BUCKET ) // alas, not there
791
+ return end();
792
+ else
793
+ return const_iterator(this,
794
+ table.get_iter(pos.first), table.nonempty_end());
795
+ }
796
+
797
+ // This is a tr1 method: the bucket a given key is in, or what bucket
798
+ // it would be put in, if it were to be inserted. Shrug.
799
+ size_type bucket(const key_type& key) const {
800
+ pair<size_type, size_type> pos = find_position(key);
801
+ return pos.first == ILLEGAL_BUCKET ? pos.second : pos.first;
802
+ }
803
+
804
+ // Counts how many elements have key key. For maps, it's either 0 or 1.
805
+ size_type count(const key_type &key) const {
806
+ pair<size_type, size_type> pos = find_position(key);
807
+ return pos.first == ILLEGAL_BUCKET ? 0 : 1;
808
+ }
809
+
810
+ // Likewise, equal_range doesn't really make sense for us. Oh well.
811
+ pair<iterator,iterator> equal_range(const key_type& key) {
812
+ iterator pos = find(key); // either an iterator or end
813
+ if (pos == end()) {
814
+ return pair<iterator,iterator>(pos, pos);
815
+ } else {
816
+ const iterator startpos = pos++;
817
+ return pair<iterator,iterator>(startpos, pos);
818
+ }
819
+ }
820
+ pair<const_iterator,const_iterator> equal_range(const key_type& key) const {
821
+ const_iterator pos = find(key); // either an iterator or end
822
+ if (pos == end()) {
823
+ return pair<const_iterator,const_iterator>(pos, pos);
824
+ } else {
825
+ const const_iterator startpos = pos++;
826
+ return pair<const_iterator,const_iterator>(startpos, pos);
827
+ }
828
+ }
829
+
830
+
831
+ // INSERTION ROUTINES
832
+ private:
833
+ // If you know *this is big enough to hold obj, use this routine
834
+ pair<iterator, bool> insert_noresize(const value_type& obj) {
835
+ // First, double-check we're not inserting delkey
836
+ assert(!use_deleted || !equals(get_key(obj), delkey));
837
+ const pair<size_type,size_type> pos = find_position(get_key(obj));
838
+ if ( pos.first != ILLEGAL_BUCKET) { // object was already there
839
+ return pair<iterator,bool>(iterator(this, table.get_iter(pos.first),
840
+ table.nonempty_end()),
841
+ false); // false: we didn't insert
842
+ } else { // pos.second says where to put it
843
+ if ( test_deleted(pos.second) ) { // just replace if it's been del.
844
+ // The set() below will undelete this object. We just worry about stats
845
+ assert(num_deleted > 0);
846
+ --num_deleted; // used to be, now it isn't
847
+ }
848
+ table.set(pos.second, obj);
849
+ return pair<iterator,bool>(iterator(this, table.get_iter(pos.second),
850
+ table.nonempty_end()),
851
+ true); // true: we did insert
852
+ }
853
+ }
854
+
855
+ public:
856
+ // This is the normal insert routine, used by the outside world
857
+ pair<iterator, bool> insert(const value_type& obj) {
858
+ resize_delta(1); // adding an object, grow if need be
859
+ return insert_noresize(obj);
860
+ }
861
+
862
+ // When inserting a lot at a time, we specialize on the type of iterator
863
+ template <class InputIterator>
864
+ void insert(InputIterator f, InputIterator l) {
865
+ // specializes on iterator type
866
+ insert(f, l, typename STL_NAMESPACE::iterator_traits<InputIterator>::iterator_category());
867
+ }
868
+
869
+ // Iterator supports operator-, resize before inserting
870
+ template <class ForwardIterator>
871
+ void insert(ForwardIterator f, ForwardIterator l,
872
+ STL_NAMESPACE::forward_iterator_tag) {
873
+ size_type n = STL_NAMESPACE::distance(f, l); // TODO(csilvers): standard?
874
+ resize_delta(n);
875
+ for ( ; n > 0; --n, ++f)
876
+ insert_noresize(*f);
877
+ }
878
+
879
+ // Arbitrary iterator, can't tell how much to resize
880
+ template <class InputIterator>
881
+ void insert(InputIterator f, InputIterator l,
882
+ STL_NAMESPACE::input_iterator_tag) {
883
+ for ( ; f != l; ++f)
884
+ insert(*f);
885
+ }
886
+
887
+
888
+ // DELETION ROUTINES
889
+ size_type erase(const key_type& key) {
890
+ // First, double-check we're not erasing delkey
891
+ assert(!use_deleted || !equals(key, delkey));
892
+ const_iterator pos = find(key); // shrug: shouldn't need to be const
893
+ if ( pos != end() ) {
894
+ assert(!test_deleted(pos)); // or find() shouldn't have returned it
895
+ set_deleted(pos);
896
+ ++num_deleted;
897
+ consider_shrink = true; // will think about shrink after next insert
898
+ return 1; // because we deleted one thing
899
+ } else {
900
+ return 0; // because we deleted nothing
901
+ }
902
+ }
903
+
904
+ // This is really evil: really it should be iterator, not const_iterator.
905
+ // But...the only reason keys are const is to allow lookup.
906
+ // Since that's a moot issue for deleted keys, we allow const_iterators
907
+ void erase(const_iterator pos) {
908
+ if ( pos == end() ) return; // sanity check
909
+ if ( set_deleted(pos) ) { // true if object has been newly deleted
910
+ ++num_deleted;
911
+ consider_shrink = true; // will think about shrink after next insert
912
+ }
913
+ }
914
+
915
+ void erase(const_iterator f, const_iterator l) {
916
+ for ( ; f != l; ++f) {
917
+ if ( set_deleted(f) ) // should always be true
918
+ ++num_deleted;
919
+ }
920
+ consider_shrink = true; // will think about shrink after next insert
921
+ }
922
+
923
+
924
+ // COMPARISON
925
+ bool operator==(const sparse_hashtable& ht) const {
926
+ // We really want to check that the hash functions are the same
927
+ // but alas there's no way to do this. We just hope.
928
+ return ( num_deleted == ht.num_deleted && table == ht.table );
929
+ }
930
+ bool operator!=(const sparse_hashtable& ht) const {
931
+ return !(*this == ht);
932
+ }
933
+
934
+
935
+ // I/O
936
+ // We support reading and writing hashtables to disk. NOTE that
937
+ // this only stores the hashtable metadata, not the stuff you've
938
+ // actually put in the hashtable! Alas, since I don't know how to
939
+ // write a hasher or key_equal, you have to make sure everything
940
+ // but the table is the same. We compact before writing.
941
+ bool write_metadata(FILE *fp) {
942
+ squash_deleted(); // so we don't have to worry about delkey
943
+ return table.write_metadata(fp);
944
+ }
945
+
946
+ bool read_metadata(FILE *fp) {
947
+ num_deleted = 0; // since we got rid before writing
948
+ bool result = table.read_metadata(fp);
949
+ reset_thresholds();
950
+ return result;
951
+ }
952
+
953
+ // Only meaningful if value_type is a POD.
954
+ bool write_nopointer_data(FILE *fp) {
955
+ return table.write_nopointer_data(fp);
956
+ }
957
+
958
+ // Only meaningful if value_type is a POD.
959
+ bool read_nopointer_data(FILE *fp) {
960
+ return table.read_nopointer_data(fp);
961
+ }
962
+
963
+ private:
964
+ // The actual data
965
+ hasher hash; // required by hashed_associative_container
966
+ key_equal equals;
967
+ ExtractKey get_key;
968
+ SetKey set_key;
969
+ size_type num_deleted; // how many occupied buckets are marked deleted
970
+ bool use_deleted; // false until delkey has been set
971
+ // TODO(csilvers): make a pointer, and get rid of use_deleted (benchmark!)
972
+ key_type delkey; // which key marks deleted entries
973
+ float enlarge_resize_percent; // how full before resize
974
+ float shrink_resize_percent; // how empty before resize
975
+ size_type shrink_threshold; // table.size() * shrink_resize_percent
976
+ size_type enlarge_threshold; // table.size() * enlarge_resize_percent
977
+ sparsetable<value_type> table; // holds num_buckets and num_elements too
978
+ bool consider_shrink; // true if we should try to shrink before next insert
979
+
980
+ void reset_thresholds() {
981
+ enlarge_threshold = static_cast<size_type>(table.size()
982
+ * enlarge_resize_percent);
983
+ shrink_threshold = static_cast<size_type>(table.size()
984
+ * shrink_resize_percent);
985
+ consider_shrink = false; // whatever caused us to reset already considered
986
+ }
987
+ };
988
+
989
+ // We need a global swap as well
990
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
991
+ inline void swap(sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> &x,
992
+ sparse_hashtable<V,K,HF,ExK,SetK,EqK,A> &y) {
993
+ x.swap(y);
994
+ }
995
+
996
+ #undef JUMP_
997
+
998
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
999
+ const typename sparse_hashtable<V,K,HF,ExK,SetK,EqK,A>::size_type
1000
+ sparse_hashtable<V,K,HF,ExK,SetK,EqK,A>::ILLEGAL_BUCKET;
1001
+
1002
+ // How full we let the table get before we resize. Knuth says .8 is
1003
+ // good -- higher causes us to probe too much, though saves memory
1004
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
1005
+ const float sparse_hashtable<V,K,HF,ExK,SetK,EqK,A>::HT_OCCUPANCY_FLT = 0.8f;
1006
+
1007
+ // How empty we let the table get before we resize lower.
1008
+ // It should be less than OCCUPANCY_FLT / 2 or we thrash resizing
1009
+ template <class V, class K, class HF, class ExK, class SetK, class EqK, class A>
1010
+ const float sparse_hashtable<V,K,HF,ExK,SetK,EqK,A>::HT_EMPTY_FLT = 0.4f *
1011
+ sparse_hashtable<V,K,HF,ExK,SetK,EqK,A>::HT_OCCUPANCY_FLT;
1012
+
1013
+ _END_GOOGLE_NAMESPACE_
1014
+
1015
+ #endif /* _SPARSEHASHTABLE_H_ */