turbonss/cxxmph/mph_map.h

266 lines
9.5 KiB
C
Raw Normal View History

2011-12-05 20:03:10 +02:00
#ifndef __CXXMPH_MPH_MAP_H__
#define __CXXMPH_MPH_MAP_H__
2011-11-05 19:15:11 +02:00
// Implementation of the unordered associative mapping interface using a
// minimal perfect hash function.
//
2012-06-03 10:17:14 +03:00
// Since these are header-mostly libraries, make sure you compile your code
// with -DNDEBUG and -O3. The code requires a modern C++11 compiler.
//
2012-06-03 10:17:14 +03:00
// The container comes in 3 flavors, all in the cxxmph namespace and drop-in
// replacement for the popular classes with the same names.
// * dense_hash_map
// -> fast, uses more memory, 2.93 bits per bucket, ~50% occupation
// * unordered_map (aliases: hash_map, mph_map)
// -> middle ground, uses 2.93 bits per bucket, ~81% occupation
// * sparse_hash_map -> slower, uses 3.6 bits per bucket
// -> less fast, uses 3.6 bits per bucket, 100% occupation
//
// Those classes are not necessarily faster than their existing counterparts.
// Benchmark your code before using it. The larger the key, the larger the
// number of elements inserted, and the bigger the number of failed searches,
// the more likely those classes will outperform existing code.
2012-04-12 22:36:23 +03:00
//
2012-06-03 10:17:14 +03:00
// For large sets of urls (>100k), which are a somewhat expensive to compare, I
// found those class to be about 10%-50% faster than unordered_map.
2011-11-05 19:15:11 +02:00
2010-11-08 22:19:44 +02:00
#include <algorithm>
#include <iostream>
#include <limits>
2011-11-10 20:44:37 +02:00
#include <unordered_map>
#include <unordered_set>
2010-06-28 22:01:18 +03:00
#include <vector>
#include <utility> // for std::pair
2012-05-08 20:22:47 +03:00
#include "hollow_iterator.h"
2012-03-14 23:26:26 +02:00
#include "mph_bits.h"
#include "mph_index.h"
2012-05-08 20:22:47 +03:00
#include "seeded_hash.h"
2010-10-29 09:26:37 +03:00
namespace cxxmph {
2011-12-10 03:57:37 +02:00
using std::pair;
using std::make_pair;
using std::vector;
2011-05-24 03:18:24 +03:00
2010-06-28 22:01:18 +03:00
// Save on repetitive typing.
2012-06-03 09:13:06 +03:00
#define MPH_MAP_TMPL_SPEC \
template <bool minimal, bool square, \
class Key, class Data, class HashFcn, class EqualKey, class Alloc>
#define MPH_MAP_CLASS_SPEC mph_map_base<minimal, square, Key, Data, HashFcn, EqualKey, Alloc>
2011-05-16 02:47:42 +03:00
#define MPH_MAP_METHOD_DECL(r, m) MPH_MAP_TMPL_SPEC typename MPH_MAP_CLASS_SPEC::r MPH_MAP_CLASS_SPEC::m
2012-04-15 06:03:00 +03:00
#define MPH_MAP_INLINE_METHOD_DECL(r, m) MPH_MAP_TMPL_SPEC inline typename MPH_MAP_CLASS_SPEC::r MPH_MAP_CLASS_SPEC::m
2010-06-28 22:01:18 +03:00
2012-06-03 09:13:06 +03:00
template <bool minimal, bool square, class Key, class Data, class HashFcn = std::hash<Key>, class EqualKey = std::equal_to<Key>, class Alloc = std::allocator<Data> >
class mph_map_base {
2010-06-28 22:01:18 +03:00
public:
typedef Key key_type;
typedef Data data_type;
2011-12-10 03:57:37 +02:00
typedef pair<Key, Data> value_type;
2010-06-28 22:01:18 +03:00
typedef HashFcn hasher;
typedef EqualKey key_equal;
typedef typename vector<value_type>::pointer pointer;
typedef typename vector<value_type>::reference reference;
typedef typename vector<value_type>::const_reference const_reference;
typedef typename vector<value_type>::size_type size_type;
typedef typename vector<value_type>::difference_type difference_type;
2012-03-07 10:10:29 +02:00
typedef is_empty<const vector<value_type>> is_empty_type;
typedef hollow_iterator_base<typename vector<value_type>::iterator, is_empty_type> iterator;
typedef hollow_iterator_base<typename vector<value_type>::const_iterator, is_empty_type> const_iterator;
2010-06-28 22:01:18 +03:00
// For making macros simpler.
typedef void void_type;
typedef bool bool_type;
2011-12-10 03:57:37 +02:00
typedef pair<iterator, bool> insert_return_type;
2010-06-28 22:01:18 +03:00
2012-06-03 09:13:06 +03:00
mph_map_base();
~mph_map_base();
2010-06-28 22:01:18 +03:00
iterator begin();
iterator end();
const_iterator begin() const;
const_iterator end() const;
size_type size() const;
bool empty() const;
void clear();
void erase(iterator pos);
void erase(const key_type& k);
2011-12-10 03:57:37 +02:00
pair<iterator, bool> insert(const value_type& x);
2012-04-15 06:03:00 +03:00
inline iterator find(const key_type& k);
inline const_iterator find(const key_type& k) const;
2011-12-10 03:57:37 +02:00
typedef int32_t my_int32_t; // help macros
2012-04-15 06:03:00 +03:00
inline int32_t index(const key_type& k) const;
2010-06-28 22:01:18 +03:00
data_type& operator[](const key_type &k);
2011-06-14 08:24:40 +03:00
const data_type& operator[](const key_type &k) const;
2010-06-28 22:01:18 +03:00
2012-06-03 09:13:06 +03:00
size_type bucket_count() const { return index_.size() + slack_.bucket_count(); }
void rehash(size_type nbuckets /*ignored*/);
2010-06-28 22:01:18 +03:00
2011-06-14 08:24:40 +03:00
protected: // mimicking STL implementation
EqualKey equal_;
2010-06-28 22:01:18 +03:00
private:
2010-11-05 08:40:15 +02:00
template <typename iterator>
struct iterator_first : public iterator {
iterator_first(iterator it) : iterator(it) { }
const typename iterator::value_type::first_type& operator*() {
2010-11-05 08:40:15 +02:00
return this->iterator::operator*().first;
}
};
template <typename iterator>
iterator_first<iterator> make_iterator_first(iterator it) {
return iterator_first<iterator>(it);
}
2011-05-24 03:18:24 +03:00
void pack();
vector<value_type> values_;
vector<bool> present_;
2012-06-03 09:13:06 +03:00
FlexibleMPHIndex<minimal, square, Key, typename seeded_hash<HashFcn>::hash_function> index_;
2012-03-21 15:20:30 +02:00
// TODO(davi) optimize slack to use hash from index rather than calculate its own
2012-06-03 09:13:06 +03:00
typedef std::unordered_map<h128, uint32_t, h128::hash32> slack_type;
2010-11-05 08:40:15 +02:00
slack_type slack_;
size_type size_;
2012-03-21 15:20:30 +02:00
typename seeded_hash<HashFcn>::hash_function hasher128_;
2010-06-28 22:01:18 +03:00
};
2011-05-16 02:47:42 +03:00
MPH_MAP_TMPL_SPEC
bool operator==(const MPH_MAP_CLASS_SPEC& lhs, const MPH_MAP_CLASS_SPEC& rhs) {
return lhs.size() == rhs.size() && std::equal(lhs.begin(), lhs.end(), rhs.begin());
2010-06-28 22:01:18 +03:00
}
2012-06-03 09:13:06 +03:00
MPH_MAP_TMPL_SPEC MPH_MAP_CLASS_SPEC::mph_map_base() : size_(0) {
clear();
2011-05-24 03:18:24 +03:00
pack();
2010-06-28 22:01:18 +03:00
}
2012-06-03 09:13:06 +03:00
MPH_MAP_TMPL_SPEC MPH_MAP_CLASS_SPEC::~mph_map_base() { }
2010-06-28 22:01:18 +03:00
2011-05-16 02:47:42 +03:00
MPH_MAP_METHOD_DECL(insert_return_type, insert)(const value_type& x) {
auto it = find(x.first);
auto it_end = end();
if (it != it_end) return make_pair(it, false);
bool should_pack = false;
2012-03-07 10:10:29 +02:00
if (values_.capacity() == values_.size() && values_.size() > 256) {
should_pack = true;
}
2010-06-28 22:01:18 +03:00
values_.push_back(x);
present_.push_back(true);
++size_;
2012-03-21 15:20:30 +02:00
h128 h = hasher128_.hash128(x.first, 0);
if (slack_.find(h) != slack_.end()) should_pack = true; // unavoidable pack
else slack_.insert(std::make_pair(h, values_.size() - 1));
2012-03-07 10:10:29 +02:00
if (should_pack) pack();
2010-06-28 22:01:18 +03:00
it = find(x.first);
2011-12-10 03:57:37 +02:00
return make_pair(it, true);
2010-06-28 22:01:18 +03:00
}
2011-05-24 03:18:24 +03:00
MPH_MAP_METHOD_DECL(void_type, pack)() {
// fprintf(stderr, "Paki %d values\n", values_.size());
2010-06-28 22:01:18 +03:00
if (values_.empty()) return;
assert(std::unordered_set<key_type>(make_iterator_first(begin()), make_iterator_first(end())).size() == size());
bool success = index_.Reset(
make_iterator_first(begin()),
make_iterator_first(end()), size_);
2012-04-15 06:03:00 +03:00
if (!success) { exit(-1); }
2012-06-03 09:13:06 +03:00
vector<value_type> new_values(index_.size());
new_values.reserve(new_values.size() * 2);
2012-06-03 09:13:06 +03:00
vector<bool> new_present(index_.size(), false);
new_present.reserve(new_present.size() * 2);
for (iterator it = begin(), it_end = end(); it != it_end; ++it) {
2012-06-03 09:13:06 +03:00
size_type id = index_.index(it->first);
assert(id < index_.size());
2011-05-16 05:04:30 +03:00
assert(id < new_values.size());
2010-11-06 13:14:07 +02:00
new_values[id] = *it;
new_present[id] = true;
2010-06-28 22:01:18 +03:00
}
2012-03-14 23:26:26 +02:00
// fprintf(stderr, "Collision ratio: %f\n", collisions*1.0/size());
2010-06-28 22:01:18 +03:00
values_.swap(new_values);
present_.swap(new_present);
slack_type().swap(slack_);
2010-06-28 22:01:18 +03:00
}
MPH_MAP_METHOD_DECL(iterator, begin)() { return make_hollow(&values_, &present_, values_.begin()); }
MPH_MAP_METHOD_DECL(iterator, end)() { return make_solid(&values_, &present_, values_.end()); }
MPH_MAP_METHOD_DECL(const_iterator, begin)() const { return make_hollow(&values_, &present_, values_.begin()); }
MPH_MAP_METHOD_DECL(const_iterator, end)() const { return make_solid(&values_, &present_, values_.end()); }
MPH_MAP_METHOD_DECL(bool_type, empty)() const { return size_ == 0; }
MPH_MAP_METHOD_DECL(size_type, size)() const { return size_; }
2010-06-28 22:01:18 +03:00
2011-05-16 02:47:42 +03:00
MPH_MAP_METHOD_DECL(void_type, clear)() {
2010-06-28 22:01:18 +03:00
values_.clear();
present_.clear();
2010-06-28 22:01:18 +03:00
slack_.clear();
index_.clear();
size_ = 0;
2010-06-28 22:01:18 +03:00
}
2011-05-16 02:47:42 +03:00
MPH_MAP_METHOD_DECL(void_type, erase)(iterator pos) {
present_[pos.it_ - begin().it_] = false;
*pos = value_type();
--size_;
2010-06-28 22:01:18 +03:00
}
2011-05-16 02:47:42 +03:00
MPH_MAP_METHOD_DECL(void_type, erase)(const key_type& k) {
2010-06-28 22:01:18 +03:00
iterator it = find(k);
if (it == end()) return;
erase(it);
}
2012-04-15 06:03:00 +03:00
MPH_MAP_INLINE_METHOD_DECL(const_iterator, find)(const key_type& k) const {
auto idx = index(k);
typename vector<value_type>::const_iterator vit = values_.begin() + idx;
if (idx == -1 || vit->first != k) return end();
return make_solid(&values_, &present_, vit);;
2010-06-28 22:01:18 +03:00
}
2011-06-14 08:24:40 +03:00
2012-04-15 06:03:00 +03:00
MPH_MAP_INLINE_METHOD_DECL(iterator, find)(const key_type& k) {
2012-04-14 23:59:15 +03:00
auto idx = index(k);
typename vector<value_type>::iterator vit = values_.begin() + idx;
if (idx == -1 || vit->first != k) return end();
return make_solid(&values_, &present_, vit);;
2012-04-14 23:59:15 +03:00
}
MPH_MAP_INLINE_METHOD_DECL(my_int32_t, index)(const key_type& k) const {
2012-04-22 03:50:14 +03:00
if (__builtin_expect(!slack_.empty(), 0)) {
auto sit = slack_.find(hasher128_.hash128(k, 0));
if (sit != slack_.end()) return sit->second;
}
2012-06-03 09:13:06 +03:00
if (__builtin_expect(index_.size(), 1)) {
auto id = index_.index(k);
if (__builtin_expect(present_[id], true)) {
return id;
}
}
2012-04-14 23:59:15 +03:00
return -1;
2011-06-14 08:24:40 +03:00
}
2011-05-16 02:47:42 +03:00
MPH_MAP_METHOD_DECL(data_type&, operator[])(const key_type& k) {
2011-12-10 03:57:37 +02:00
return insert(make_pair(k, data_type())).first->second;
2010-06-28 22:01:18 +03:00
}
MPH_MAP_METHOD_DECL(void_type, rehash)(size_type nbuckets) {
pack();
vector<value_type>(values_.begin(), values_.end()).swap(values_);
vector<bool>(present_.begin(), present_.end()).swap(present_);
slack_type().swap(slack_);
}
2012-06-03 09:13:06 +03:00
#define MPH_MAP_PREAMBLE template <class Key, class Data,\
class HashFcn = std::hash<Key>, class EqualKey = std::equal_to<Key>,\
class Alloc = std::allocator<Data> >
MPH_MAP_PREAMBLE class mph_map : public mph_map_base<
false, false, Key, Data, HashFcn, EqualKey, Alloc> {};
MPH_MAP_PREAMBLE class unordered_map : public mph_map_base<
false, false, Key, Data, HashFcn, EqualKey, Alloc> {};
MPH_MAP_PREAMBLE class hash_map : public mph_map_base<
false, false, Key, Data, HashFcn, EqualKey, Alloc> {};
MPH_MAP_PREAMBLE class dense_hash_map : public mph_map_base<
false, true, Key, Data, HashFcn, EqualKey, Alloc> {};
MPH_MAP_PREAMBLE class sparse_hash_map : public mph_map_base<
true, false, Key, Data, HashFcn, EqualKey, Alloc> {};
2010-10-29 09:26:37 +03:00
} // namespace cxxmph
2011-12-05 20:03:10 +02:00
#endif // __CXXMPH_MPH_MAP_H__