This page introduces into the stxxl::map (for further information on the structure you may have a look at Map (B+-tree)).
stxxl::map is an external associative container that stores elements formed by a combination of a unique key value and a data value, following a specific order. The map's key values are generally used to sort and uniquely identify the data values, while the data values store the content associated to this key.
Creating a STXXL Map
To create a stxxl::map object, several template parameters are required. The first two parameters KeyType and DataType which is an std::pair<int, char> in this example are self-explanatory, the third parameter has to be a comparator class which is used to determine whether a key is smaller than another one, the fourth and fifth parameter define the node- and leaf block size.
#define DATA_NODE_BLOCK_SIZE (4096)
#define DATA_LEAF_BLOCK_SIZE (4096)
...
map_type my_map((map_type::node_block_type::raw_size) * 3, (map_type::leaf_block_type::raw_size) * 3);
The comparator class has to be defined by hand (and before the map definition above) and looks like:
struct ComparatorGreater
{
bool operator () (const int & a, const int & b) const
{ return a > b; }
static int max_value()
};
If CompareGreater()(a,b) is true, then a is smaller than b. CompareType must also provide a static max_value method, that returns a value of type KeyType that is larger than any key stored in map, i.e. for all x in map holds CompareType()(x,CompareType::max_value())
Naturally, we can define a comparator class which returns true if b is smaller than a as follows:
struct CompareLess
{
bool operator () (const int & a, const int & b) const
{ return a<b; }
static int max_value() const
};
Note that CompareType must define a strict weak ordering.
Insert elements
Insertion of elements is possible in three different ways:
- simple insertion
my_map.insert(std::pair<int, char>(1, 'a'));
my_map.insert(std::pair<int, char>(2, 'b'));
my_map.insert(std::pair<int, char>(3, 'c'));
my_map.insert(std::pair<int, char>(4, 'd'));
- insertion with hint
map_type::iterator iter = my_map.begin();
my_map.insert(iter, std::pair<int, char>(5, 'w'));
my_map.insert(iter, std::pair<int, char>(6, 'x'));
my_map.insert(iter, std::pair<int, char>(7, 'y'));
my_map.insert(iter, std::pair<int, char>(8, 'z'));
- range insertion
map_type anothermap((map_type::node_block_type::raw_size) * 3, (map_type::leaf_block_type::raw_size) * 3);
anothermap.insert(my_map.begin(),my_map.find('c'));
Access elements
Random access is possible by using the []-operator:
std::cout << "my_map[4] is " << my_map[4] << std::endl;
Scanning a stxxl::map by an iterator works like
for (iter = my_map.begin(); iter != my_map.end(); ++iter)
{
std::cout << iter->first << " => " << iter->second << std::endl;
}
Hint: To enable leaf prefetching during scanning, call my_map.enable_prefetching() before.
In addition, the operations lower_bound() and upper_bound() are available. The function lower_bound(key) returns an iterator which initially points to the first element in the container whose key is not considered to go before key. upper_bound(key) works similar as it returns an iterator which initially points to the first element in the container whose key is considered to go after key.
map_type::iterator iter_low, iter_up;
iter_low = my_map.lower_bound(2);
iter_up = my_map.upper_bound(6);
std::cout << "lower bound " << iter_low->second << " upper bound " << iter_up->second << std::endl;
Note: lower_bound() works nearly equal to upper_bound(), except in the case that the map contains an element with a key equivalent lower_bound(x): In this case lower_bound(x) returns an iterator pointing to that element, whereas upper_bound(x) returns an iterator pointing to the next element.
Delete elements
Removing elements out of the map is possible in three different ways:
- Erasing by iterator
map_type::iter iter = my_map.find(7);
my_map.erase(iter);
- Erasing by key
- Erasing by range
iter = my_map.find(3);
my_map.erase(iter, my_map.end());
Determine size / Check whether the map is empty
To determine the size (i.e. the number of elements) of an instance, call size():
std::cout << "number of elements in my_map: " << my_map.size() << std::endl;
To check if the priority queue is empty, call empty() which returns true in case:
std::cout << "is my_map empty? " << my_map.empty() << std::endl;
A minimal working example on STXXL Map
(See examples/containers/map1.cpp for the sourcecode of the following example).
#include <iostream>
#define DATA_NODE_BLOCK_SIZE (4096)
#define DATA_LEAF_BLOCK_SIZE (4096)
struct CompareGreater
{
bool operator () (const int& a, const int& b) const
{ return a > b; }
static int max_value()
};
int main()
{
map_type my_map((map_type::node_block_type::raw_size)*3, (map_type::leaf_block_type::raw_size)*3);
my_map.insert(std::pair<int, char>(1, 'a'));
my_map.insert(std::pair<int, char>(2, 'b'));
my_map.insert(std::pair<int, char>(3, 'c'));
my_map.insert(std::pair<int, char>(4, 'd'));
my_map.erase(3);
map_type::iterator iter;
std::cout << "my_map contains:\n";
for (iter = my_map.begin(); iter != my_map.end(); ++iter)
{
std::cout << iter->first << " => " << iter->second << std::endl;
}
map_type::iterator iter_low, iter_up;
iter_low = my_map.lower_bound(1);
iter_up = my_map.upper_bound(3);
std::cout << "lower bound " << iter_low->second << ", upper bound " << iter_up->second << std::endl;
return 0;
}