• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /////////////////////////////////////////////////////////////////////////////
2 //
3 // (C) Copyright Ion Gaztanaga  2006-2013
4 //
5 // Distributed under the Boost Software License, Version 1.0.
6 //    (See accompanying file LICENSE_1_0.txt or copy at
7 //          http://www.boost.org/LICENSE_1_0.txt)
8 //
9 // See http://www.boost.org/libs/intrusive for documentation.
10 //
11 /////////////////////////////////////////////////////////////////////////////
12 //[doc_assoc_optimized_code_normal_find
13 #include <boost/intrusive/set.hpp>
14 #include <boost/intrusive/unordered_set.hpp>
15 #include <cstring>
16 
17 using namespace boost::intrusive;
18 
19 // Hash function for strings
20 struct StrHasher
21 {
operator ()StrHasher22    std::size_t operator()(const char *str) const
23    {
24       std::size_t seed = 0;
25       for(; *str; ++str)   boost::hash_combine(seed, *str);
26       return seed;
27    }
28 };
29 
30 class Expensive : public set_base_hook<>, public unordered_set_base_hook<>
31 {
32    std::string key_;
33    // Other members...
34 
35    public:
Expensive(const char * key)36    Expensive(const char *key)
37       :  key_(key)
38       {}  //other expensive initializations...
39 
get_key() const40    const std::string & get_key() const
41       {  return key_;   }
42 
operator <(const Expensive & a,const Expensive & b)43    friend bool operator <  (const Expensive &a, const Expensive &b)
44       {  return a.key_ < b.key_;  }
45 
operator ==(const Expensive & a,const Expensive & b)46    friend bool operator == (const Expensive &a, const Expensive &b)
47       {  return a.key_ == b.key_;  }
48 
hash_value(const Expensive & object)49    friend std::size_t hash_value(const Expensive &object)
50       {  return StrHasher()(object.get_key().c_str());  }
51 };
52 
53 // A set and unordered_set that store Expensive objects
54 typedef set<Expensive>           Set;
55 typedef unordered_set<Expensive> UnorderedSet;
56 
57 // Search functions
get_from_set(const char * key,Set & set_object)58 Expensive *get_from_set(const char* key, Set &set_object)
59 {
60    Set::iterator it = set_object.find(Expensive(key));
61    if( it == set_object.end() )     return 0;
62    return &*it;
63 }
64 
get_from_uset(const char * key,UnorderedSet & uset_object)65 Expensive *get_from_uset(const char* key, UnorderedSet &uset_object)
66 {
67    UnorderedSet::iterator it = uset_object.find(Expensive (key));
68    if( it == uset_object.end() )  return 0;
69    return &*it;
70 }
71 //]
72 
73 //[doc_assoc_optimized_code_optimized_find
74 // These compare Expensive and a c-string
75 struct StrExpComp
76 {
operator ()StrExpComp77    bool operator()(const char *str, const Expensive &c) const
78    {  return std::strcmp(str, c.get_key().c_str()) < 0;  }
79 
operator ()StrExpComp80    bool operator()(const Expensive &c, const char *str) const
81    {  return std::strcmp(c.get_key().c_str(), str) < 0;  }
82 };
83 
84 struct StrExpEqual
85 {
operator ()StrExpEqual86    bool operator()(const char *str, const Expensive &c) const
87    {  return std::strcmp(str, c.get_key().c_str()) == 0;  }
88 
operator ()StrExpEqual89    bool operator()(const Expensive &c, const char *str) const
90    {  return std::strcmp(c.get_key().c_str(), str) == 0;  }
91 };
92 
93 // Optimized search functions
get_from_set_optimized(const char * key,Set & set_object)94 Expensive *get_from_set_optimized(const char* key, Set &set_object)
95 {
96    Set::iterator it = set_object.find(key, StrExpComp());
97    if( it == set_object.end() )   return 0;
98    return &*it;
99 }
100 
get_from_uset_optimized(const char * key,UnorderedSet & uset_object)101 Expensive *get_from_uset_optimized(const char* key, UnorderedSet &uset_object)
102 {
103    UnorderedSet::iterator it = uset_object.find(key, StrHasher(), StrExpEqual());
104    if( it == uset_object.end() )  return 0;
105    return &*it;
106 }
107 //]
108 
109 //[doc_assoc_optimized_code_normal_insert
110 // Insertion functions
insert_to_set(const char * key,Set & set_object)111 bool insert_to_set(const char* key, Set &set_object)
112 {
113    Expensive *pobject = new Expensive(key);
114    bool success = set_object.insert(*pobject).second;
115    if(!success)   delete pobject;
116    return success;
117 }
118 
insert_to_uset(const char * key,UnorderedSet & uset_object)119 bool insert_to_uset(const char* key, UnorderedSet &uset_object)
120 {
121    Expensive *pobject = new Expensive(key);
122    bool success = uset_object.insert(*pobject).second;
123    if(!success)   delete pobject;
124    return success;
125 }
126 //]
127 
128 //[doc_assoc_optimized_code_optimized_insert
129 // Optimized insertion functions
insert_to_set_optimized(const char * key,Set & set_object)130 bool insert_to_set_optimized(const char* key, Set &set_object)
131 {
132    Set::insert_commit_data insert_data;
133    bool success = set_object.insert_check(key, StrExpComp(), insert_data).second;
134    if(success) set_object.insert_commit(*new Expensive(key), insert_data);
135    return success;
136 }
137 
insert_to_uset_optimized(const char * key,UnorderedSet & uset_object)138 bool insert_to_uset_optimized(const char* key, UnorderedSet &uset_object)
139 {
140    UnorderedSet::insert_commit_data insert_data;
141    bool success = uset_object.insert_check
142       (key, StrHasher(), StrExpEqual(), insert_data).second;
143    if(success) uset_object.insert_commit(*new Expensive(key), insert_data);
144    return success;
145 }
146 //]
147 
main()148 int main()
149 {
150    Set set;
151    UnorderedSet::bucket_type buckets[10];
152    UnorderedSet unordered_set(UnorderedSet::bucket_traits(buckets, 10));
153 
154    const char * const expensive_key
155       = "A long string that avoids small string optimization";
156 
157    Expensive value(expensive_key);
158 
159    if(get_from_set(expensive_key, set)){
160       return 1;
161    }
162 
163    if(get_from_uset(expensive_key, unordered_set)){
164       return 1;
165    }
166 
167    if(get_from_set_optimized(expensive_key, set)){
168       return 1;
169    }
170 
171    if(get_from_uset_optimized(expensive_key, unordered_set)){
172       return 1;
173    }
174 
175    Set::iterator setit =  set.insert(value).first;
176    UnorderedSet::iterator unordered_setit =  unordered_set.insert(value).first;
177 
178    if(!get_from_set(expensive_key, set)){
179       return 1;
180    }
181 
182    if(!get_from_uset(expensive_key, unordered_set)){
183       return 1;
184    }
185 
186    if(!get_from_set_optimized(expensive_key, set)){
187       return 1;
188    }
189 
190    if(!get_from_uset_optimized(expensive_key, unordered_set)){
191       return 1;
192    }
193 
194    set.erase(setit);
195    unordered_set.erase(unordered_setit);
196 
197    if(!insert_to_set(expensive_key, set)){
198       return 1;
199    }
200 
201    if(!insert_to_uset(expensive_key, unordered_set)){
202       return 1;
203    }
204 
205    {
206       Expensive *ptr = &*set.begin();
207       set.clear();
208       delete ptr;
209    }
210 
211    {
212       Expensive *ptr = &*unordered_set.begin();
213       unordered_set.clear();
214       delete ptr;
215    }
216 
217    if(!insert_to_set_optimized(expensive_key, set)){
218       return 1;
219    }
220 
221    if(!insert_to_uset_optimized(expensive_key, unordered_set)){
222       return 1;
223    }
224 
225    {
226       Expensive *ptr = &*set.begin();
227       set.clear();
228       delete ptr;
229    }
230 
231    {
232       Expensive *ptr = &*unordered_set.begin();
233       unordered_set.clear();
234       delete ptr;
235    }
236 
237    setit       =  set.insert(value).first;
238    unordered_setit   =  unordered_set.insert(value).first;
239 
240    if(insert_to_set(expensive_key, set)){
241       return 1;
242    }
243 
244    if(insert_to_uset(expensive_key, unordered_set)){
245       return 1;
246    }
247 
248    if(insert_to_set_optimized(expensive_key, set)){
249       return 1;
250    }
251 
252    if(insert_to_uset_optimized(expensive_key, unordered_set)){
253       return 1;
254    }
255 
256    set.erase(value);
257    unordered_set.erase(value);
258 
259    return 0;
260 }
261