forked from andrewprock/ustl
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathumap.h
168 lines (157 loc) · 8.29 KB
/
umap.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
// This file is part of the uSTL library, an STL implementation.
//
// Copyright (c) 2005 by Mike Sharov <[email protected]>
// This file is free software, distributed under the MIT License.
#pragma once
#include "uvector.h"
#include "ufunction.h"
namespace ustl {
template <typename Pair, typename Comp>
struct pair_compare_first : public binary_function<Pair,Pair,bool> {
inline bool operator()(const Pair& a, const Pair& b) { return Comp()(a.first,b.first); }
};
template <typename K, typename V, typename Comp>
struct pair_compare_first_key : public binary_function<pair<K,V>,K,bool> {
inline bool operator()(const pair<K,V>& a, const K& b) { return Comp()(a.first,b); }
inline bool operator()(const K& a, const pair<K,V>& b) { return Comp()(a,b.first); }
};
/// \class map umap.h ustl.h
/// \ingroup AssociativeContainers
///
/// \brief A sorted associative container of pair<K,V>
///
template <typename K, typename V, typename Comp = less<K> >
class map : public vector<pair<K,V> > {
public:
typedef K key_type;
typedef V data_type;
typedef const K& const_key_ref;
typedef const V& const_data_ref;
typedef const map<K,V,Comp>& rcself_t;
typedef vector<pair<K,V> > base_class;
typedef typename base_class::value_type value_type;
typedef typename base_class::size_type size_type;
typedef typename base_class::pointer pointer;
typedef typename base_class::const_pointer const_pointer;
typedef typename base_class::reference reference;
typedef typename base_class::const_reference const_reference;
typedef typename base_class::const_iterator const_iterator;
typedef typename base_class::iterator iterator;
typedef typename base_class::reverse_iterator reverse_iterator;
typedef typename base_class::const_reverse_iterator const_reverse_iterator;
typedef pair<const_iterator,const_iterator> const_range_t;
typedef pair<iterator,iterator> range_t;
typedef pair<iterator,bool> insertrv_t;
typedef std::initializer_list<value_type> initlist_t;
typedef Comp key_compare;
typedef pair_compare_first<value_type,Comp> value_compare;
typedef pair_compare_first_key<K,V,Comp> value_key_compare;
public:
inline map (void) : base_class() {}
explicit inline map (size_type n) : base_class (n) {}
inline map (rcself_t v) : base_class (v) {}
inline map (const_iterator i1, const_iterator i2) : base_class() { insert (i1, i2); }
inline rcself_t operator= (rcself_t v) { base_class::operator= (v); return *this; }
inline const_data_ref at (const_key_ref k) const { assert (find(k) != end()); return find(k)->second; }
inline data_type& at (const_key_ref k) { assert (find(k) != end()); return find(k)->second; }
inline const_data_ref operator[] (const_key_ref i) const { return at(i); }
data_type& operator[] (const_key_ref i);
inline key_compare key_comp (void) const { return key_compare(); }
inline value_compare value_comp (void) const { return value_compare(); }
inline size_type size (void) const { return base_class::size(); }
inline iterator begin (void) { return base_class::begin(); }
inline const_iterator begin (void) const { return base_class::begin(); }
inline iterator end (void) { return base_class::end(); }
inline const_iterator end (void) const { return base_class::end(); }
inline void assign (const_iterator i1, const_iterator i2) { clear(); insert (i1, i2); }
inline void push_back (const_reference v) { insert (v); }
inline const_iterator find (const_key_ref k) const;
inline iterator find (const_key_ref k) { return const_cast<iterator> (const_cast<rcself_t>(*this).find (k)); }
inline const_iterator find_data (const_data_ref v, const_iterator first = nullptr, const_iterator last = nullptr) const;
inline iterator find_data (const_data_ref v, iterator first = nullptr, iterator last = nullptr) { return const_cast<iterator> (find_data (v, const_cast<const_iterator>(first), const_cast<const_iterator>(last))); }
const_iterator lower_bound (const_key_ref k) const { return ::ustl::lower_bound (begin(), end(), k, value_key_compare()); }
inline iterator lower_bound (const_key_ref k) { return const_cast<iterator>(const_cast<rcself_t>(*this).lower_bound (k)); }
const_iterator upper_bound (const_key_ref k) const { return ::ustl::upper_bound (begin(), end(), k, value_key_compare()); }
inline iterator upper_bound (const_key_ref k) { return const_cast<iterator>(const_cast<rcself_t>(*this).upper_bound (k)); }
const_range_t equal_range (const_key_ref k) const { return ::ustl::equal_range (begin(), end(), k, value_key_compare()); }
inline range_t equal_range (const_key_ref k) { return ::ustl::equal_range (begin(), end(), k, value_key_compare()); }
inline size_type count (const_key_ref v) const { const_range_t r = equal_range(v); return distance(r.first,r.second); }
insertrv_t insert (const_reference v);
inline iterator insert (const_iterator, const_reference v) { return insert(v).first; }
void insert (const_iterator i1, const_iterator i2) { for (; i1 != i2; ++i1) insert (*i1); }
inline void erase (const_key_ref k);
inline iterator erase (iterator ep) { return base_class::erase (ep); }
inline iterator erase (iterator ep1, iterator ep2) { return base_class::erase (ep1, ep2); }
inline void clear (void) { base_class::clear(); }
inline void swap (map& v) { base_class::swap (v); }
#if HAVE_CPP11
inline map (map&& v) : base_class (move(v)) {}
inline map (initlist_t v) : base_class() { insert (v.begin(), v.end()); }
inline map& operator= (map&& v) { base_class::operator= (move(v)); return *this; }
insertrv_t insert (value_type&& v);
inline iterator insert (const_iterator, value_type&& v) { return insert(move(v)).first; }
inline void insert (initlist_t v) { insert (v.begin(), v.end()); }
template <typename... Args>
inline insertrv_t emplace (Args&&... args) { return insert (value_type(forward<Args>(args)...)); }
template <typename... Args>
inline iterator emplace_hint (const_iterator h, Args&&... args) { return insert (h, value_type(forward<Args>(args)...)); }
template <typename... Args>
inline insertrv_t emplace_back (Args&&... args) { return insert (value_type(forward<Args>(args)...)); }
#endif
};
/// Returns the pair<K,V> where K = \p k.
template <typename K, typename V, typename Comp>
inline typename map<K,V,Comp>::const_iterator map<K,V,Comp>::find (const_key_ref k) const
{
const_iterator i = lower_bound (k);
return (i < end() && Comp()(k,i->first)) ? end() : i;
}
/// Returns the pair<K,V> where V = \p v, occuring in range [first,last).
template <typename K, typename V, typename Comp>
inline typename map<K,V,Comp>::const_iterator map<K,V,Comp>::find_data (const_data_ref v, const_iterator first, const_iterator last) const
{
if (!first) first = begin();
if (!last) last = end();
for (; first != last && first->second != v; ++first) ;
return first;
}
/// Returns data associated with key \p k.
template <typename K, typename V, typename Comp>
typename map<K,V,Comp>::data_type& map<K,V,Comp>::operator[] (const_key_ref k)
{
iterator ip = lower_bound (k);
if (ip == end() || Comp()(k,ip->first))
ip = base_class::insert (ip, make_pair (k, V()));
return ip->second;
}
/// Inserts the pair into the container.
template <typename K, typename V, typename Comp>
typename map<K,V,Comp>::insertrv_t map<K,V,Comp>::insert (const_reference v)
{
iterator ip = lower_bound (v.first);
bool bInserted = ip == end() || Comp()(v.first, ip->first);
if (bInserted)
ip = base_class::insert (ip, v);
return make_pair (ip, bInserted);
}
#if HAVE_CPP11
/// Inserts the pair into the container.
template <typename K, typename V, typename Comp>
typename map<K,V,Comp>::insertrv_t map<K,V,Comp>::insert (value_type&& v)
{
iterator ip = lower_bound (v.first);
bool bInserted = ip == end() || Comp()(v.first, ip->first);
if (bInserted)
ip = base_class::insert (ip, move(v));
return make_pair (ip, bInserted);
}
#endif
/// Erases the element with key value \p k.
template <typename K, typename V, typename Comp>
inline void map<K,V,Comp>::erase (const_key_ref k)
{
iterator ip = find (k);
if (ip != end())
erase (ip);
}
} // namespace ustl