Struct concurrent_map::ConcurrentMap 
source · pub struct ConcurrentMap<K, V, const FANOUT: usize = 64, const LOCAL_GC_BUFFER_SIZE: usize = 128>where
    K: 'static + Clone + Minimum + Ord + Send + Sync,
    V: 'static + Clone + Send + Sync,{ /* private fields */ }Expand description
A lock-free B+ tree.
Note that this structure is Send but NOT Sync,
despite being a lock-free tree. This is because the
inner reclamation system, provided by the ebr crate
completely avoids atomic operations in its hot path
for efficiency. If you want to share ConcurrentMap
between threads, simply clone it, and this will set up
a new efficient thread-local memory reclamation state.
If you want to use a custom key type, you must
implement the Minimum trait,
allowing the left-most side of the tree to be
created before inserting any data. If you wish
to perform scans in reverse lexicographical order,
you may instead implement Maximum for your key
type and use std::cmp::Reverse.
The FANOUT const generic must be greater than 3.
This const generic controls how large the fixed-size
array for either child pointers (for index nodes) or
values (for leaf nodes) will be. A higher value may
make reads and scans faster, but writes will be slower
because each modification performs a read-copy-update
of the full node. In some cases, it may be preferable
to wrap complex values in an Arc to avoid higher
copy costs.
The LOCAL_GC_BUFFER_SIZE const generic must be greater than 0.
This controls the epoch-based reclamation granularity.
Garbage is placed into fixed-size arrays, and garbage collection
only happens after this array fills up and a final timestamp is
assigned to it. Lower values will cause replaced values to be dropped
more quickly, but the efficiency will be lower. Values that are
extremely high may cause undesirable memory usage because it will
take more time to fill up each thread-local garbage segment.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
// insert and remove atomically returns the last value, if it was set,
// similarly to a BTreeMap
assert_eq!(tree.insert(1, 10), None);
assert_eq!(tree.insert(1, 11), Some(10));
assert_eq!(tree.remove(&1), Some(11));
// get also functions similarly to BTreeMap, except it
// returns a cloned version of the value rather than a
// reference to it, so that no locks need to be maintained.
// For this reason, it can be a good idea to use types that
// are cheap to clone for values, which can be easily handled
// with `Arc` etc...
assert_eq!(tree.insert(1, 12), None);
assert_eq!(tree.get(&1), Some(12));
// compare and swap from value 12 to value 20
tree.cas(1, Some(&12_usize), Some(20)).unwrap();
assert_eq!(tree.get(&1).unwrap(), 20);
// there are a lot of methods that are not covered
// here - check out the docs!Implementations§
source§impl<K, V, const FANOUT: usize, const LOCAL_GC_BUFFER_SIZE: usize> ConcurrentMap<K, V, FANOUT, LOCAL_GC_BUFFER_SIZE>where
    K: 'static + Clone + Minimum + Ord + Send + Sync,
    V: 'static + Clone + Send + Sync,
 
impl<K, V, const FANOUT: usize, const LOCAL_GC_BUFFER_SIZE: usize> ConcurrentMap<K, V, FANOUT, LOCAL_GC_BUFFER_SIZE>where K: 'static + Clone + Minimum + Ord + Send + Sync, V: 'static + Clone + Send + Sync,
sourcepub fn get<Q>(&self, key: &Q) -> Option<V>where
    K: Borrow<Q>,
    Q: Ord + ?Sized,
 
pub fn get<Q>(&self, key: &Q) -> Option<V>where K: Borrow<Q>, Q: Ord + ?Sized,
Atomically get a value out of the tree that is associated with this key.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
let actual = tree.get(&0);
let expected = None;
assert_eq!(expected, actual);
let actual = tree.get(&1);
let expected = Some(1);
assert_eq!(expected, actual);sourcepub fn get_lt(&self, key: &K) -> Option<(K, V)>
 
pub fn get_lt(&self, key: &K) -> Option<(K, V)>
Atomically get a key and value out of the tree that is associated with the key that is lexicographically less than the provided key.
This will always return None if the key passed to get_lt == K::MIN.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
let actual = tree.get_lt(&0);
let expected = None;
assert_eq!(expected, actual);
let actual = tree.get_lt(&1);
let expected = None;
assert_eq!(expected, actual);
let actual = tree.get_lt(&2);
let expected = Some((1, 1));
assert_eq!(expected, actual);sourcepub fn get_lte(&self, key: &K) -> Option<(K, V)>
 
pub fn get_lte(&self, key: &K) -> Option<(K, V)>
Atomically get a key and value out of the tree that is associated with the key that is lexicographically less than or equal to the provided key.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
let actual = tree.get_lte(&0);
let expected = None;
assert_eq!(expected, actual);
let actual = tree.get_lte(&1);
let expected = Some((1, 1));
assert_eq!(expected, actual);
let actual = tree.get_lte(&2);
let expected = Some((1, 1));
assert_eq!(expected, actual);sourcepub fn get_gt(&self, key: &K) -> Option<(K, V)>
 
pub fn get_gt(&self, key: &K) -> Option<(K, V)>
Atomically get a key and value out of the tree that is associated with the key that is lexicographically greater than the provided key.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
let actual = tree.get_gt(&0);
let expected = Some((1, 1));
assert_eq!(expected, actual);
let actual = tree.get_gt(&1);
let expected = None;
assert_eq!(expected, actual);
let actual = tree.get_gt(&2);
let expected = None;
assert_eq!(expected, actual);sourcepub fn get_gte(&self, key: &K) -> Option<(K, V)>
 
pub fn get_gte(&self, key: &K) -> Option<(K, V)>
Atomically get a key and value out of the tree that is associated with the key that is lexicographically greater than or equal to the provided key.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
let actual = tree.get_gte(&0);
let expected = Some((1, 1));
assert_eq!(expected, actual);
let actual = tree.get_gte(&1);
let expected = Some((1, 1));
assert_eq!(expected, actual);
let actual = tree.get_gte(&2);
let expected = None;
assert_eq!(expected, actual);sourcepub fn first(&self) -> Option<(K, V)>
 
pub fn first(&self) -> Option<(K, V)>
Get the minimum item stored in this structure.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
tree.insert(2, 2);
tree.insert(3, 3);
let actual = tree.first();
let expected = Some((1, 1));
assert_eq!(actual, expected);sourcepub fn pop_first(&self) -> Option<(K, V)>where
    V: PartialEq,
 
pub fn pop_first(&self) -> Option<(K, V)>where V: PartialEq,
Atomically remove the minimum item stored in this structure.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
tree.insert(2, 2);
tree.insert(3, 3);
let actual = tree.pop_first();
let expected = Some((1, 1));
assert_eq!(actual, expected);
assert_eq!(tree.get(&1), None);sourcepub fn last(&self) -> Option<(K, V)>
 
pub fn last(&self) -> Option<(K, V)>
Get the maximum item stored in this structure.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
tree.insert(2, 2);
tree.insert(3, 3);
let actual = tree.last();
let expected = Some((3, 3));
assert_eq!(actual, expected);sourcepub fn pop_last(&self) -> Option<(K, V)>where
    V: PartialEq,
 
pub fn pop_last(&self) -> Option<(K, V)>where V: PartialEq,
Atomically remove the maximum item stored in this structure.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
tree.insert(1, 1);
tree.insert(2, 2);
tree.insert(3, 3);
let actual = tree.pop_last();
let expected = Some((3, 3));
assert_eq!(actual, expected);
assert_eq!(tree.get(&3), None);sourcepub fn insert(&self, key: K, value: V) -> Option<V>
 
pub fn insert(&self, key: K, value: V) -> Option<V>
Atomically insert a key-value pair into the tree, returning the previous value associated with this key if one existed.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
assert_eq!(tree.insert(1, 1), None);
assert_eq!(tree.insert(1, 1), Some(1));sourcepub fn remove<Q>(&self, key: &Q) -> Option<V>where
    K: Borrow<Q>,
    Q: Ord + ?Sized,
 
pub fn remove<Q>(&self, key: &Q) -> Option<V>where K: Borrow<Q>, Q: Ord + ?Sized,
Atomically remove the value associated with this key from the tree, returning the previous value if one existed.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
assert_eq!(tree.remove(&1), None);
assert_eq!(tree.insert(1, 1), None);
assert_eq!(tree.remove(&1), Some(1));sourcepub fn cas<VRef>(
    &self,
    key: K,
    old: Option<&VRef>,
    new: Option<V>
) -> Result<Option<V>, CasFailure<V>>where
    V: Borrow<VRef>,
    VRef: PartialEq + ?Sized,
 
pub fn cas<VRef>( &self, key: K, old: Option<&VRef>, new: Option<V> ) -> Result<Option<V>, CasFailure<V>>where V: Borrow<VRef>, VRef: PartialEq + ?Sized,
Atomically compare and swap the value associated with this key from the old value to the
new one. An old value of None means “only create this value if it does not already
exist”. A new value of None means “delete this value, if it matches the provided old value”.
If successful, returns the old value if it existed. If unsuccessful, returns the proposed
new value.
Examples
let tree = concurrent_map::ConcurrentMap::<usize, usize>::default();
// key 1 does not yet exist
assert_eq!(tree.get(&1), None);
// uniquely create value 10
tree.cas(1, None, Some(10)).unwrap();
assert_eq!(tree.get(&1).unwrap(), 10);
// compare and swap from value 10 to value 20
tree.cas(1, Some(&10_usize), Some(20)).unwrap();
assert_eq!(tree.get(&1).unwrap(), 20);
// if we guess the wrong current value, a CasFailure is returned
// which will tell us what the actual current value is (which we
// failed to provide) and it will give us back our proposed new
// value.
let cas_result = tree.cas(1, Some(&999999_usize), Some(30));
let expected_cas_failure = Err(concurrent_map::CasFailure {
    actual: Some(20),
    returned_new_value: Some(30),
});
assert_eq!(cas_result, expected_cas_failure);
// conditionally delete
tree.cas(1, Some(&20_usize), None).unwrap();
assert_eq!(tree.get(&1), None);sourcepub fn len(&self) -> usize
 
pub fn len(&self) -> usize
A lagging, eventually-consistent length count. This is NOT atomically
updated with [insert] / [remove] / [cas], but is updated after those
operations complete their atomic modifications to the shared map.
sourcepub fn is_empty(&self) -> bool
 
pub fn is_empty(&self) -> bool
A lagging, eventually-consistent check for emptiness, based on the correspondingly
non-atomic len method.
sourcepub fn iter(&self) -> Iter<'_, K, V, FANOUT, LOCAL_GC_BUFFER_SIZE> ⓘ
 
pub fn iter(&self) -> Iter<'_, K, V, FANOUT, LOCAL_GC_BUFFER_SIZE> ⓘ
Iterate over the tree.
This is not an atomic snapshot, and it caches B+tree leaf nodes as it iterates through them to achieve high throughput. As a result, the following behaviors are possible:
- may (or may not!) return values that were concurrently added to the tree after the iterator was created
- may (or may not!) return items that were concurrently deleted from the tree after the iterator was created
- If a key’s value is changed from one value to another one after this iterator is created, this iterator might return the old or the new value.
But, you can be certain that any key that existed prior to the creation of this iterator, and was not changed during iteration, will be observed as expected.
sourcepub fn range<R: RangeBounds<K>>(
    &self,
    range: R
) -> Iter<'_, K, V, FANOUT, LOCAL_GC_BUFFER_SIZE, R> ⓘ
 
pub fn range<R: RangeBounds<K>>( &self, range: R ) -> Iter<'_, K, V, FANOUT, LOCAL_GC_BUFFER_SIZE, R> ⓘ
Iterate over a range of the tree.
This is not an atomic snapshot, and it caches B+tree leaf nodes as it iterates through them to achieve high throughput. As a result, the following behaviors are possible:
- may (or may not!) return values that were concurrently added to the tree after the iterator was created
- may (or may not!) return items that were concurrently deleted from the tree after the iterator was created
- If a key’s value is changed from one value to another one after this iterator is created, this iterator might return the old or the new value.
But, you can be certain that any key that existed prior to the creation of this iterator, and was not changed during iteration, will be observed as expected.
Panics
This will panic if the provided range’s end_bound() == Bound::Excluded(K::MIN).
Trait Implementations§
source§impl<K, V, const FANOUT: usize, const LOCAL_GC_BUFFER_SIZE: usize> Clone for ConcurrentMap<K, V, FANOUT, LOCAL_GC_BUFFER_SIZE>where
    K: 'static + Clone + Minimum + Ord + Send + Sync + Clone,
    V: 'static + Clone + Send + Sync + Clone,
 
impl<K, V, const FANOUT: usize, const LOCAL_GC_BUFFER_SIZE: usize> Clone for ConcurrentMap<K, V, FANOUT, LOCAL_GC_BUFFER_SIZE>where K: 'static + Clone + Minimum + Ord + Send + Sync + Clone, V: 'static + Clone + Send + Sync + Clone,
source§fn clone(&self) -> ConcurrentMap<K, V, FANOUT, LOCAL_GC_BUFFER_SIZE>
 
fn clone(&self) -> ConcurrentMap<K, V, FANOUT, LOCAL_GC_BUFFER_SIZE>
1.0.0 · source§fn clone_from(&mut self, source: &Self)
 
fn clone_from(&mut self, source: &Self)
source. Read more