QuickCppLib 0.10
Eliminate all the tedious hassle when making state-of-the-art C++ 14 - 23 libraries!
Loading...
Searching...
No Matches
quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir > Class Template Reference

Never-allocating in-place bitwise Fredkin trie index head type. More...

#include "bitwise_trie.hpp"

Inheritance diagram for quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >:

Public Types

using key_type = decltype(_item_accessors(static_cast< ItemType * >(nullptr)).key())
 Key type indexing the items.
 
using mapped_type = ItemType *
 The type of item indexed.
 
using value_type = ItemType *
 The value type.
 
using size_type = decltype(bitwise_trie_head_accessors< const Base, const ItemType >(nullptr).size())
 The size type.
 
using difference_type = ptrdiff_t
 The type of a difference between pointers to the type of item indexed.
 
using reference = ItemType &
 A reference to the type of item indexed.
 
using const_reference = const ItemType
 A const reference to the type of item indexed.
 
using pointer = ItemType *
 A pointer to the type of item indexed.
 
using const_pointer = const ItemType *
 A const pointer to the type of item indexed.
 
using iterator = iterator_< false, bitwise_trie, pointer, reference >
 The iterator type.
 
using const_iterator = iterator_< true, const bitwise_trie, const_pointer, const_reference >
 The const iterator type.
 
using reverse_iterator = std::reverse_iterator< iterator >
 The reverse iterator type.
 
using const_reverse_iterator = std::reverse_iterator< const_iterator >
 The const reverse iterator type.
 

Public Member Functions

void triecheckvalidity (const key_type *tocheck=nullptr) const noexcept
 
constexpr bitwise_trie ()
 
template<class Arg , class... Args, typename std::enable_if<(std::is_constructible< Base, Arg, Args... >::value), bool >::type = true>
constexpr bitwise_trie (Arg &&arg, Args &&...args)
 
 bitwise_trie (const bitwise_trie &o) noexcept
 
bitwise_trieoperator= (const bitwise_trie &o) noexcept
 
void swap (bitwise_trie &o) noexcept
 Swaps the contents of the index.
 
QUICKCPPLIB_NODISCARD constexpr bool empty () const noexcept
 True if the bitwise trie is empty.
 
constexpr size_type size () const noexcept
 Returns the number of items in the bitwise trie.
 
constexpr size_type max_size () const noexcept
 Returns the maximum number of items in the index.
 
reference front () noexcept
 Returns the front of the index.
 
const_reference front () const noexcept
 Returns the front of the index.
 
reference back () noexcept
 Returns the back of the index.
 
const_reference back () const noexcept
 Returns the back of the index.
 
iterator begin () noexcept
 Returns an iterator to the first item in the index.
 
const_iterator begin () const noexcept
 Returns an iterator to the first item in the index.
 
reverse_iterator rbegin () noexcept
 Returns an iterator to the last item in the index.
 
const_reverse_iterator rbegin () const noexcept
 Returns an iterator to the last item in the index.
 
const_iterator cbegin () const noexcept
 Returns an iterator to the first item in the index.
 
const_reverse_iterator crbegin () const noexcept
 Returns an iterator to the last item in the index.
 
iterator end () noexcept
 Returns an iterator to the item after the last in the index.
 
const_iterator end () const noexcept
 Returns an iterator to the item before the first in the index.
 
reverse_iterator rend () noexcept
 Returns an iterator to the item before the first in the index.
 
const_reverse_iterator rend () const noexcept
 Returns an iterator to the item before the first in the index.
 
const_iterator cend () const noexcept
 Returns an iterator to the item after the last in the index.
 
const_reverse_iterator crend () const noexcept
 Returns an iterator to the item before the first in the index.
 
constexpr void clear () noexcept
 Clears the index.
 
size_type count (key_type k) const noexcept
 Return how many items with key there are.
 
size_type count (const_iterator it) const noexcept
 Return how many items with the same key as iterator there are.
 
iterator insert (pointer p)
 
iterator erase (const_iterator it) noexcept
 Erases an item.
 
void erase (pointer p) noexcept
 Erases an item.
 
iterator erase (key_type k) noexcept
 Erases an item.
 
iterator find (key_type k) const noexcept
 Finds an item.
 
iterator find_equal_or_larger (key_type k, int64_t rounds) const noexcept
 
iterator find_equal_or_next_largest (key_type k) const noexcept
 
iterator upper_bound (key_type k) const noexcept
 Finds the item next larger than the key. This is equivalent to find_equal_or_next_largest(k + 1).
 
iterator upper_bound_estimate (key_type k, int64_t rounds) const noexcept
 Estimate the item next larger than the key. This is equivalent to find_equal_or_larger(k + 1, rounds).
 
bool contains (key_type k) const noexcept
 True if the index contains the key.
 
reference operator[] (key_type k) noexcept
 Returns a reference to the specified element, aborting if key not found.
 
const_reference operator[] (const key_type &k) const noexcept
 Returns a reference to the specified element, aborting if key not found.
 

Static Public Attributes

static constexpr int nobble_direction = (NobbleDir < 0) ? -1 : ((NobbleDir > 0) ? 1 : 0)
 The direction of nobble configured.
 

Detailed Description

template<class Base, class ItemType, int NobbleDir = 0>
class quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >

Never-allocating in-place bitwise Fredkin trie index head type.

Template Parameters
BaseThe base type from which to inherit (and thus overlay the index member functions).
ItemTypeThe type of item indexed.
NobbleDir-1 to nobble zeros, +1 to nobble ones, 0 to nobble both equally (see below).

This uses the bitwise Fredkin trie algorithm to index a collection of items by an unsigned integral key (e.g. a size_t from std::hash), providing identical O(log2 N) time insertion, removal, and finds (i.e. insert, remove and find all take identical time). It is thus most like a red-black tree. However it has a number of very useful characteristics which make it invaluable in certain use cases.

Firstly, unlike a hash table, this algorithm requires no additional memory allocation whatsoever. It wholly and exclusively uses only the items added to the index, and the index head, for storage. This makes it invaluable for bootstrap type scenarios, such as in memory allocators or first boot of a kernel. It also works well with C++ exceptions globally disabled.

Secondly, unlike a hash table it provides a bounded time close fit find, which is particularly useful for where you need an item closely matching what you need, but you don't care if it's the closest matching item. An example of where this is super useful to have is in memory allocators, where you need a free block bigger than or equal to the size you are allocating.

There is also a guaranteed closest rather than closely matching item find, however it is more expensive than a red-black tree whose guaranteed sortedness makes finding exact upper bounds easy.

Thirdly, unlike any other algorithm, this one is much faster for low key values than large key values, so if you can keep most of your keys small, you will see more benefit.

As you can see, bitwise tries have most of the same benefits of red-black trees, but they approximate hash tables for performance of insert-find-remove-findclosefit on most CPUs. This makes them very compelling where red-black trees are too slow, or where some concurrency is desirable (concurrent modify is easy to implement for all keys whose topmost set bit differs).

The order of the items during iteration is somewhat sorted by key incrementing. Note the somewhat, it is fully ordered for each top bit set increment, but within each of those there is an interleave of insertion order and key increment. A bubble sort can be a good sort algorithm depending on relationship of key increment to insertion order, but you shouldn't assume it to be so without empirically checking.

Items inserted with the same key preserve order of insertion. Performance is great on all CPUs which have a single cycle opcode for finding the first set bit in a key. If your CPU has a slow bitscan opcode (e.g. older Intel Atom), performance is merely good rather than great.

The index is intrusive, as in, your types must provide the storage needed by the index for housekeeping. You can very tightly pack or compress or calculate those numbers by defining a specialised bitwise_trie_head_accessors<Base, ItemType> if you wish to customise storage of housekeeping for the trie index head; and/or specialise bitwise_trie_item_accessors<ItemType> if you wish to customise storage of housekeeping for each item indexed by the trie. The default implementations of those accessor types require various member variables prefixed with trie_ in your types:

  • The default bitwise_trie_head_accessors<Base, ItemType> requires the following member variables in the trie index head type:
    • <unsigned type> trie_count
    • ItemType *trie_children[8 * sizeof(<unsigned type>)]
    • bool trie_nobbledir (if you use equal nobbling only)
  • The default bitwise_trie_item_accessors<ItemType> requires the following member variables in the trie item type:
    • ItemType *trie_parent
    • ItemType *trie_child[2]
    • ItemType *trie_sibling[2] (if you allow multiple items with the same key value only)
    • KeyType trie_key

Again, I stress that the above can be completely customised and packed tighter with custom accessor type specialisations for your type. The tigher you can pack your structures, the more fits into L3 cache, and the faster everything goes. You can also store these in a file, and store offset pointers which are safe when a memory map relocates in memory.

Close and closest fit finds always find an item whose key is larger or equal to the key sought. Lower bound is not implemented yet.

Most of this implementation is lifted from https://github.com/ned14/nedtries, but it has been modernised for current C++ idomatic practice.

Differences from a C++ container

As this is an index, not a container, the value type is always a pointer. I chose pointers instead of references to aid readability i.e. it is very clear from reading code using this index that it is an index not a container.

Because custom accessors may not store pointers as pointers, reference is not a reference, but also a pointer. Iteration thus yields pointers, not references.

Nobble direction

During item removal only, to keep the tree balanced one needs to choose which node to nobble. If for all the keys you use there is a surplus of zeros after the first set bit (this would be common for pointers), you should nobble zeros by setting the template parameter NobbleDir to -1. If for all the keys you use there is a surplus of ones after the first set bit, you should nobble ones by setting the template parameter NobbleDir to 1.

If for all the keys you use there is an equal balance between zeros and ones after the first set bit (this would be common for hashes), you will get the best results if you add state storage to keep a nobble direction boolean which flips between false and true such that nobbling is equally distributed. In this case, set NobbleDir to 0.

tl;dr; If your key results from a hash function, choose NobbleDir = 0. If your key results from a pointer, or is some number which clusters on regular even boundaries, choose NobbleDir = -1.

Todo:
Implement lower_bound().

Member Typedef Documentation

◆ key_type

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::key_type = decltype(_item_accessors(static_cast<ItemType *>(nullptr)).key())

Key type indexing the items.

◆ mapped_type

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::mapped_type = ItemType *

The type of item indexed.

◆ value_type

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::value_type = ItemType *

The value type.

◆ size_type

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::size_type = decltype(bitwise_trie_head_accessors<const Base, const ItemType>(nullptr).size())

The size type.

◆ difference_type

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::difference_type = ptrdiff_t

The type of a difference between pointers to the type of item indexed.

◆ reference

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::reference = ItemType &

A reference to the type of item indexed.

◆ const_reference

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::const_reference = const ItemType

A const reference to the type of item indexed.

◆ pointer

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::pointer = ItemType *

A pointer to the type of item indexed.

◆ const_pointer

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::const_pointer = const ItemType *

A const pointer to the type of item indexed.

◆ iterator

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::iterator = iterator_<false, bitwise_trie, pointer, reference>

The iterator type.

◆ const_iterator

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::const_iterator = iterator_<true, const bitwise_trie, const_pointer, const_reference>

The const iterator type.

◆ reverse_iterator

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::reverse_iterator = std::reverse_iterator<iterator>

The reverse iterator type.

◆ const_reverse_iterator

template<class Base , class ItemType , int NobbleDir = 0>
using quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::const_reverse_iterator = std::reverse_iterator<const_iterator>

The const reverse iterator type.

Constructor & Destructor Documentation

◆ bitwise_trie() [1/3]

template<class Base , class ItemType , int NobbleDir = 0>
constexpr quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::bitwise_trie ( )
inlineconstexpr
1775{ clear(); }
constexpr void clear() noexcept
Clears the index.
Definition bitwise_trie.hpp:1919

◆ bitwise_trie() [2/3]

template<class Base , class ItemType , int NobbleDir = 0>
template<class Arg , class... Args, typename std::enable_if<(std::is_constructible< Base, Arg, Args... >::value), bool >::type = true>
constexpr quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::bitwise_trie ( Arg &&  arg,
Args &&...  args 
)
inlineexplicitconstexpr
1779 : Base(static_cast<Arg &&>(arg), static_cast<Args &&>(args)...)
1780 {
1781 clear();
1782 }

◆ bitwise_trie() [3/3]

template<class Base , class ItemType , int NobbleDir = 0>
quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::bitwise_trie ( const bitwise_trie< Base, ItemType, NobbleDir > &  o)
inlinenoexcept
1784 {
1785 auto myhead = _head_accessors();
1786 auto ohead = o._head_accessors();
1787 for(unsigned n = 0; n < _key_type_bits; n++)
1788 {
1789 myhead.set_child(n, ohead.child(n));
1790 }
1791 myhead.set_size(ohead.size());
1792 }

Member Function Documentation

◆ triecheckvalidity()

template<class Base , class ItemType , int NobbleDir = 0>
void quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::triecheckvalidity ( const key_type tocheck = nullptr) const
inlinenoexcept
1499 {
1500 (void) tocheck;
1501#ifndef NDEBUG
1502 auto head = _head_accessors();
1503 const_pointer node = nullptr, child = nullptr;
1504 _trie_validity_state state;
1505 memset(&state, 0, sizeof(state));
1506 for(unsigned n = 0; n < _key_type_bits; n++)
1507 {
1508 if(tocheck != nullptr)
1509 {
1510 if((((key_type) -1 << n) & *tocheck) != ((key_type) 1 << n))
1511 {
1512 continue;
1513 }
1514 }
1515 if((node = head.child(n)) != nullptr)
1516 {
1517 auto nodelink = _item_accessors(node);
1518 key_type nodekey = nodelink.key();
1519 _lock_unlock_branch lock_unlock(this, nodekey, false, n);
1520 state.tops++;
1521 auto bitidx = nodelink.bit_index();
1522 assert(bitidx == n);
1523 assert(head.child(bitidx) == node);
1524 assert(0 == nodekey || ((((size_t) -1) << bitidx) & nodekey) == ((size_t) 1 << bitidx));
1525 while(_item_accessors(child = nodelink.sibling(true)).is_secondary_sibling())
1526 {
1527 state.leafs++;
1528 auto childlink = _item_accessors(child);
1529 assert(nullptr == childlink.parent());
1530 assert(nullptr == childlink.child(false));
1531 assert(nullptr == childlink.child(true));
1532 assert(child == _item_accessors(childlink.sibling(false)).sibling(true));
1533 assert(child == _item_accessors(childlink.sibling(true)).sibling(false));
1534 nodelink = childlink;
1535 state.count++;
1536 }
1537 nodelink = _item_accessors(node);
1538 state.count++;
1539 if(nodelink.child(0) != nullptr)
1540 {
1541 state.lefts++;
1542 state.smallestkey = (key_type) -1;
1543 state.largestkey = 0;
1544 _triecheckvaliditybranch(nodelink.child(0), bitidx - 1, state);
1545 assert(0 == state.smallestkey || state.smallestkey >= (key_type) 1 << bitidx);
1546 assert(bitidx + 1 >= _key_type_bits || state.largestkey < (key_type) 1 << (bitidx + 1));
1547 }
1548 if(nodelink.child(1) != nullptr)
1549 {
1550 state.rights++;
1551 state.smallestkey = (key_type) -1;
1552 state.largestkey = 0;
1553 _triecheckvaliditybranch(nodelink.child(1), bitidx - 1, state);
1554 assert(state.smallestkey >= (key_type) 1 << bitidx);
1555 assert(bitidx + 1 >= _key_type_bits || state.largestkey < (key_type) 1 << (bitidx + 1));
1556 }
1557 }
1558 }
1559 if(tocheck == nullptr)
1560 {
1561 assert(state.count == head.size());
1562 }
1563#if 1
1564 for(state.count = 0, node = _triemin(); node != nullptr; (node = _trienext(node)), state.count++)
1565#if 0
1566 printf("%p\n", node)
1567#endif
1568 ;
1569 if(state.count != head.size())
1570 {
1571 assert(state.count == head.size());
1572 }
1573#if 0
1574 printf("\n");
1575#endif
1576 for(state.count = 0, node = _triemax(); node != nullptr; (node = _trieprev(node)), state.count++)
1577#if 0
1578 printf("%p\n", node)
1579#endif
1580 ;
1581 if(state.count != head.size())
1582 {
1583 assert(state.count == head.size());
1584 }
1585#if 0
1586 printf("\n");
1587#endif
1588#if !defined(NDEBUG) && 0
1589 if(count > 50)
1590 printf("Of count %u, tops %.2lf%%, lefts %.2lf%%, rights %.2lf%%, leafs %.2lf%%\n", count,
1591 100.0 * tops / count, 100.0 * lefts / count, 100.0 * rights / count, 100.0 * leafs / count);
1592#endif
1593#endif /* !NDEBUG */
1594#endif
1595 }
decltype(_item_accessors(static_cast< ItemType * >(nullptr)).key()) key_type
Key type indexing the items.
Definition bitwise_trie.hpp:520
size_type count(key_type k) const noexcept
Return how many items with key there are.
Definition bitwise_trie.hpp:1929
const ItemType * const_pointer
A const pointer to the type of item indexed.
Definition bitwise_trie.hpp:536

◆ operator=()

template<class Base , class ItemType , int NobbleDir = 0>
bitwise_trie & quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::operator= ( const bitwise_trie< Base, ItemType, NobbleDir > &  o)
inlinenoexcept
1794 {
1795 auto myhead = _head_accessors();
1796 auto ohead = o._head_accessors();
1797 for(unsigned n = 0; n < _key_type_bits; n++)
1798 {
1799 myhead.set_child(n, ohead.child(n));
1800 }
1801 myhead.set_size(ohead.size());
1802 return *this;
1803 }

◆ swap()

template<class Base , class ItemType , int NobbleDir = 0>
void quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::swap ( bitwise_trie< Base, ItemType, NobbleDir > &  o)
inlinenoexcept

Swaps the contents of the index.

1807 {
1808 auto myhead = _head_accessors();
1809 auto ohead = o._head_accessors();
1810 for(unsigned n = 0; n < _key_type_bits; n++)
1811 {
1812 auto t = myhead.child(n);
1813 myhead.set_child(n, ohead.child(n));
1814 ohead.set_child(n, t);
1815 }
1816 auto t = myhead.size();
1817 myhead.set_size(ohead.size());
1818 ohead.set_size(t);
1819 }

◆ empty()

template<class Base , class ItemType , int NobbleDir = 0>
QUICKCPPLIB_NODISCARD constexpr bool quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::empty ( ) const
inlineconstexprnoexcept

True if the bitwise trie is empty.

1822{ return size() == 0; }
constexpr size_type size() const noexcept
Returns the number of items in the bitwise trie.
Definition bitwise_trie.hpp:1824

◆ size()

template<class Base , class ItemType , int NobbleDir = 0>
constexpr size_type quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::size ( ) const
inlineconstexprnoexcept

Returns the number of items in the bitwise trie.

1824{ return _head_accessors().size(); }
constexpr _index_type size() const noexcept
Definition bitwise_trie.hpp:344

◆ max_size()

template<class Base , class ItemType , int NobbleDir = 0>
constexpr size_type quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::max_size ( ) const
inlineconstexprnoexcept

Returns the maximum number of items in the index.

1826{ return _head_accessors().max_size(); }
constexpr _index_type max_size() const noexcept
Definition bitwise_trie.hpp:349

◆ front() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
reference quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::front ( )
inlinenoexcept

Returns the front of the index.

1830 {
1831 if(auto p = _triemin())
1832 {
1833 return *p;
1834 }
1835 abort();
1836 }

◆ front() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
const_reference quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::front ( ) const
inlinenoexcept

Returns the front of the index.

1839 {
1840 if(auto p = _triemin())
1841 {
1842 return *p;
1843 }
1844 abort();
1845 }

◆ back() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
reference quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::back ( )
inlinenoexcept

Returns the back of the index.

1848 {
1849 if(auto p = _triemax())
1850 {
1851 return *p;
1852 }
1853 abort();
1854 }

◆ back() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
const_reference quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::back ( ) const
inlinenoexcept

Returns the back of the index.

1857 {
1858 if(auto p = _triemax())
1859 {
1860 return *p;
1861 }
1862 abort();
1863 }

◆ begin() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::begin ( )
inlinenoexcept

Returns an iterator to the first item in the index.

1867 {
1868 if(auto p = _triemin())
1869 {
1870 return iterator(this, p);
1871 }
1872 return iterator(this);
1873 }
iterator_< false, bitwise_trie, pointer, reference > iterator
The iterator type.
Definition bitwise_trie.hpp:1760

◆ begin() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
const_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::begin ( ) const
inlinenoexcept

Returns an iterator to the first item in the index.

1876 {
1877 if(auto p = _triemin())
1878 {
1879 return const_iterator(this, p);
1880 }
1881 return const_iterator(this);
1882 }
iterator_< true, const bitwise_trie, const_pointer, const_reference > const_iterator
The const iterator type.
Definition bitwise_trie.hpp:1762

◆ rbegin() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
reverse_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::rbegin ( )
inlinenoexcept

Returns an iterator to the last item in the index.

1885 {
1886 if(auto p = _triemax())
1887 {
1888 return reverse_iterator(iterator(this, p));
1889 }
1890 return reverse_iterator(iterator(this));
1891 }
std::reverse_iterator< iterator > reverse_iterator
The reverse iterator type.
Definition bitwise_trie.hpp:1764

◆ rbegin() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
const_reverse_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::rbegin ( ) const
inlinenoexcept

Returns an iterator to the last item in the index.

1894 {
1895 if(auto p = _triemax())
1896 {
1897 return const_reverse_iterator(const_iterator(this, p));
1898 }
1900 }
std::reverse_iterator< const_iterator > const_reverse_iterator
The const reverse iterator type.
Definition bitwise_trie.hpp:1766

◆ cbegin()

template<class Base , class ItemType , int NobbleDir = 0>
const_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::cbegin ( ) const
inlinenoexcept

Returns an iterator to the first item in the index.

1902{ return begin(); }
iterator begin() noexcept
Returns an iterator to the first item in the index.
Definition bitwise_trie.hpp:1866

◆ crbegin()

template<class Base , class ItemType , int NobbleDir = 0>
const_reverse_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::crbegin ( ) const
inlinenoexcept

Returns an iterator to the last item in the index.

1904{ return rbegin(); }
reverse_iterator rbegin() noexcept
Returns an iterator to the last item in the index.
Definition bitwise_trie.hpp:1884

◆ end() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::end ( )
inlinenoexcept

Returns an iterator to the item after the last in the index.

1906{ return iterator(this); }

◆ end() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
const_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::end ( ) const
inlinenoexcept

Returns an iterator to the item before the first in the index.

1908{ return const_iterator(this); }

◆ rend() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
reverse_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::rend ( )
inlinenoexcept

Returns an iterator to the item before the first in the index.

1910{ return reverse_iterator(iterator(this)); }

◆ rend() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
const_reverse_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::rend ( ) const
inlinenoexcept

Returns an iterator to the item before the first in the index.

1912{ return const_reverse_iterator(const_iterator(this)); }

◆ cend()

template<class Base , class ItemType , int NobbleDir = 0>
const_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::cend ( ) const
inlinenoexcept

Returns an iterator to the item after the last in the index.

1914{ return const_iterator(this); }

◆ crend()

template<class Base , class ItemType , int NobbleDir = 0>
const_reverse_iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::crend ( ) const
inlinenoexcept

Returns an iterator to the item before the first in the index.

1916{ return const_reverse_iterator(const_iterator(this)); }

◆ clear()

template<class Base , class ItemType , int NobbleDir = 0>
constexpr void quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::clear ( )
inlineconstexprnoexcept

Clears the index.

1920 {
1921 auto head = _head_accessors();
1922 for(unsigned n = 0; n < _key_type_bits; n++)
1923 {
1924 head.set_child(n, nullptr);
1925 }
1926 head.set_size(0);
1927 }

◆ count() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
size_type quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::count ( key_type  k) const
inlinenoexcept

Return how many items with key there are.

1929{ return count(find(k)); }
iterator find(key_type k) const noexcept
Finds an item.
Definition bitwise_trie.hpp:1988

◆ count() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
size_type quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::count ( const_iterator  it) const
inlinenoexcept

Return how many items with the same key as iterator there are.

1932 {
1933 if(it != end())
1934 {
1935 size_type ret = 1;
1936 auto plink = _item_accessors(it._p);
1937 for(auto *i = plink.sibling(true); i != it._p; i = _item_accessors(i).sibling(true))
1938 {
1939 ret++;
1940 }
1941 return ret;
1942 }
1943 return 0;
1944 }
decltype(bitwise_trie_head_accessors< const Base, const ItemType >(nullptr).size()) size_type
The size type.
Definition bitwise_trie.hpp:526
iterator end() noexcept
Returns an iterator to the item after the last in the index.
Definition bitwise_trie.hpp:1906

◆ insert()

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::insert ( pointer  p)
inline

Inserts a new item, returning an iterator to the new item if the key is new. If there is an item with that key already, if sibling storage is enabled, insert a new item with the same key. If sibling storage is disabled, return an iterator to the existing item.

If the maximum number of items has been inserted, if C++ exceptions are disabled or QUICKCPPLIB_ALGORITHM_BITWISE_TRIE_DISABLE_EXCEPTION_THROWS != 0, the returned iterator is invalid if there is no more space. Otherwise it throws std::length_error.

1954 {
1955 if(size() == max_size())
1956 {
1957#if __cpp_exceptions && !QUICKCPPLIB_ALGORITHM_BITWISE_TRIE_DISABLE_EXCEPTION_THROWS
1958 throw std::length_error("too many items");
1959#else
1960 return end();
1961#endif
1962 }
1963 p = _trieinsert(p);
1964 if(p != nullptr)
1965 {
1966 return iterator(this, p);
1967 }
1968 return end();
1969 }
constexpr size_type max_size() const noexcept
Returns the maximum number of items in the index.
Definition bitwise_trie.hpp:1826

◆ erase() [1/3]

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::erase ( const_iterator  it)
inlinenoexcept

Erases an item.

1972 {
1973 if(it == end())
1974 {
1975 assert(it != end());
1976 return end();
1977 }
1978 iterator ret(this, const_cast<pointer>(it._p));
1979 ++ret;
1980 _trieremove(const_cast<pointer>(it._p));
1981 return ret;
1982 }
ItemType * pointer
A pointer to the type of item indexed.
Definition bitwise_trie.hpp:534

◆ erase() [2/3]

template<class Base , class ItemType , int NobbleDir = 0>
void quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::erase ( pointer  p)
inlinenoexcept

Erases an item.

1984{ _trieremove(p); }

◆ erase() [3/3]

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::erase ( key_type  k)
inlinenoexcept

Erases an item.

1986{ return erase(find(k)); }
iterator erase(const_iterator it) noexcept
Erases an item.
Definition bitwise_trie.hpp:1971

◆ find()

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::find ( key_type  k) const
inlinenoexcept

Finds an item.

1989 {
1990 if(auto p = _triefind(k))
1991 {
1992 return iterator(this, p);
1993 }
1994 return iterator(this);
1995 }

◆ find_equal_or_larger()

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::find_equal_or_larger ( key_type  k,
int64_t  rounds 
) const
inlinenoexcept

Finds either an item with identical key, or an item with a larger key. The higher the value in rounds, the less average distance between the larger key and the key requested. The complexity of this function is bound by rounds.

Some useful statistics about rounds values for five million ideally distributed keys:

  • rounds = 17 returns an item with key 99.99% close to the ideal key.
  • rounds = 12 returns an item with key 99.9% close to the ideal key.
  • rounds = 7 returns an item with key 99.1% close to the ideal key.
  • rounds = 3 returns an item with key 95.1% close to the ideal key.
  • rounds = 2 returns an item with key 92% close to the ideal key.
2009 {
2010 if(auto p = _trieCfind(k, rounds))
2011 {
2012 return iterator(this, p);
2013 }
2014 return iterator(this);
2015 }

◆ find_equal_or_next_largest()

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::find_equal_or_next_largest ( key_type  k) const
inlinenoexcept

Finds either an item with identical key, or an item with the guaranteed next largest key. This is identical to close_find(k, INT64_MAX) and its average case complexity is O(log N) where N is the number of items in the index with the same top bit set. This is equivalent to upper_bound(k - 1).

2020 {
2021 if(auto p = _trieCfind(k, INT64_MAX))
2022 {
2023 return iterator(this, p);
2024 }
2025 return iterator(this);
2026 }

◆ upper_bound()

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::upper_bound ( key_type  k) const
inlinenoexcept

Finds the item next larger than the key. This is equivalent to find_equal_or_next_largest(k + 1).

2029 {
2030 if(auto p = _trieCfind(k + 1, INT64_MAX))
2031 {
2032 return iterator(this, p);
2033 }
2034 return iterator(this);
2035 }

◆ upper_bound_estimate()

template<class Base , class ItemType , int NobbleDir = 0>
iterator quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::upper_bound_estimate ( key_type  k,
int64_t  rounds 
) const
inlinenoexcept

Estimate the item next larger than the key. This is equivalent to find_equal_or_larger(k + 1, rounds).

2038 {
2039 if(auto p = _trieCfind(k + 1, rounds))
2040 {
2041 return iterator(this, p);
2042 }
2043 return iterator(this);
2044 }

◆ contains()

template<class Base , class ItemType , int NobbleDir = 0>
bool quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::contains ( key_type  k) const
inlinenoexcept

True if the index contains the key.

2046{ return nullptr != _triefind(k); }

◆ operator[]() [1/2]

template<class Base , class ItemType , int NobbleDir = 0>
reference quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::operator[] ( key_type  k)
inlinenoexcept

Returns a reference to the specified element, aborting if key not found.

2049 {
2050 if(auto p = _triefind(k))
2051 {
2052 return *p;
2053 }
2054 abort();
2055 }

◆ operator[]() [2/2]

template<class Base , class ItemType , int NobbleDir = 0>
const_reference quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::operator[] ( const key_type k) const
inlinenoexcept

Returns a reference to the specified element, aborting if key not found.

2058 {
2059 if(auto p = _triefind(k))
2060 {
2061 return *p;
2062 }
2063 abort();
2064 }

Member Data Documentation

◆ nobble_direction

template<class Base , class ItemType , int NobbleDir = 0>
constexpr int quickcpplib::_xxx::algorithm::bitwise_trie::bitwise_trie< Base, ItemType, NobbleDir >::nobble_direction = (NobbleDir < 0) ? -1 : ((NobbleDir > 0) ? 1 : 0)
staticconstexpr

The direction of nobble configured.


The documentation for this class was generated from the following file: