Policy Based Data Structure in CPP
Check References for more thorough understanding.
Policy Tree
Following is the definition of the class tree
in PBDS.
template<
typename Key,
typename Mapped,
typename Cmp_Fn = std::less<Key>,
typename Tag = rb_tree_tag,
template<
typename Const_Node_Iterator,
typename Node_Iterator,
typename Cmp_Fn_,
typename Allocator_>
class Node_Update = null_tree_node_update,
typename Allocator = std::allocator<char> >
class tree;
If you initialize the template with only the first two types, we obtain almost exact copy of the container map
.
This container can also serve as set
, for which you just need to specify the second argument template type as null_type
(in older versions it is null_mapped_type
).
Tag
— class denoting a tree structure to be used.
There are three base-classes provided in STL for use, rb_tree_tag
(red-black tree), splay_tree_tag
(splay tree) and ov_tree_tag
(ordered-vector tree).
Sadly, at competitions we can use only red-black trees because splay tree and OV-tree are using linear-timed split operation that prevents us to use them.
Node_Update
— class denoting policy for updating node invariants. By default it is set to null_node_update
, i.e., additional information is not stored in the vertices.
C++ implemented an update policy tree_order_statistics_node_update
, which stores the information required to perform order_of_key
and find_by_order
efficiently among other things.
Best way to define the class for our use in contests is as follows:
#include <ext/pb_ds/assoc_container.hpp>
using namespace __gnu_pbds;
typedef tree<int, null_type, less<int>, rb_tree_tag, tree_order_statistics_node_update> ordered_set;
// order_of_key find_by_order
typedef tree<int, int, less<int>, rb_tree_tag, tree_order_statistics_node_update> ordered_map;
This container supports all the methods supported by std::set
and we get two new methods as below.
order_of_key(val)
: returns the number of items in a set that are strictly smaller than our itemfind_by_order()
: returns an iterator to the k-th largest element (counting from zero)
Multimap and Multiset
This container can provide multimap
and multiset
in two different wayy.
Use less_equal
as Cmp_Fn
typedef tree<int, null_type, less_equal<int>, rb_tree_tag, tree_order_statistics_node_update> ordered_multiset;
If you use it with less_equal
, keep following things in mind:
_GLIBCXX_DEBUG
must not be defined, otherwise some internal check will fail.1find
will always returnend
.lower_bound
works likeupper_bound
in normal set (to return the first element > it)upper_bound
works likelower_bound
in normal set (to return the first element >= it)find_by_order
andorder_of_key
works properly (unlike the 2 functions above).
Use pair<int, int>
to store
// Main idea is to keep pairs like {elem, id}.
typedef tree< pair<int, int>, null_type, less<pair<int, int>>, rb_tree_tag, tree_order_statistics_node_update> ordered_set;
int t = 0;
ordered_set me;
me.insert({x, t++});
me.erase(me.lower_bound({x, 0}));
cout << me.order_of_key({x, 0}) << "\n";
Hash Tables
The PBDS also contain some Hash tables which are faster than unordered_set
for various reason.2
#include <ext/pb_ds/assoc_container.hpp>
using namespace __gnu_pbds;
struct chash {
const int RANDOM = (long long)(make_unique<char>().get()) ^ chrono::high_resolution_clock::now().time_since_epoch().count();
static unsigned long long hash_f(unsigned long long x) {
x += 0x9e3779b97f4a7c15;
x = (x ^ (x >> 30)) * 0xbf58476d1ce4e5b9;
x = (x ^ (x >> 27)) * 0x94d049bb133111eb;
return x ^ (x >> 31);
}
static unsigned hash_combine(unsigned a, unsigned b) { return a * 31 + b; }
int operator()(int x) const { return hash_f(x)^RANDOM; }
};
gp_hash_table<int, int, chash> table; // Hash table, good alternative of unordered_map
You will need to use Custom Hash function to not getting hacked by anti-hash tests.3 Read the Codeforces blog from the References section for Comparision of performance.
References
- Installing PBDS on MacOS
- Implicit cartesian tree in GNU C++ STL.
- C++ STL: Policy based data structures4
- GNU Library Manual Entry on Policy Based Data Structure GNU Library Docs for PBDS
- C++ STL: Policy based data structures. Part 25
- C++ STL: Order of magnitude faster hash tables with Policy Based Data Structures
-
I haven’t faced this myself, so I don’t know how reliable is this. Keeping this here for future reference ↩
-
Read Blowing up unordered_map, and how to stop getting hacked on it for more details. ↩
-
Contains links to more resources if you want to read more. ↩
-
Talks about custom
Node_Update
class for the trees. Might be useful when you want to not implement the entire tree on your own and just want change the metadata that is being captured on each node. Nice thing to keep on your back pocket, might not be directly useful for the Competitve Programming, but might be useful on more generic code. ↩