Crates.io | voluntary-servitude |
lib.rs | voluntary-servitude |
version | 4.0.8 |
source | src |
created_at | 2018-08-22 04:07:45.777724 |
updated_at | 2019-04-25 04:31:59.183752 |
description | Thread-safe appendable list with lock-free iterator |
homepage | |
repository | https://github.com/paulocsanz/voluntary-servitude |
max_upload_size | |
id | 80683 |
size | 108,046 |
Atomic abstractions (Atomic
, AtomicOption
, FillOnceAtomicOption
, FillOnceAtomicArc
)
Thread-safe appendable list with a lock-free iterator (VoluntaryServitude
- also called VS
)
Serde serialization (serde-traits
feature)
par_extend
, from_par_iter
rayon implementation (rayon-traits
feature)
Logging (logs
feature)
You probably only need this if you are debugging this crate
Box<T>
Option<Box<T>>
Option<Box<T>>
that can give references (ideal for iterators)Option<Arc<T>>
with a limited API (like FillOnceAtomicOption
)With Atomic
and AtomicOption
it's not safe to get a reference, you must replace the value to access it.
To safely get a reference of T you must use FillOnceAtomicOption
and accept the API limitations (initially None
but can be filled once).
For a safe AtomicArc
you must use some data-structure from arc-swap
, RwLock/Mutex
from parking_lot
(or std
, which is slower but the standard) or FillOnceAtomicArc
and accept the limited API (2018).
MIT and Apache-2.0
VoluntaryServitude
Examplesuse voluntary_servitude::vs;
fn main() {
let (a, b, c) = (0usize, 1usize, 2usize);
// VS alias to VoluntaryServitude
// vs! alias to voluntary_servitude! (and operates like vec!)
let list = vs![a, b, c];
assert_eq!(list.iter().collect::<Vec<_>>(), vec![&a, &b, &c]);
// Current VS's length
// Be careful with race conditions since the value, when used, may not be true anymore
assert_eq!(list.len(), 3);
// The 'iter' method makes a lock-free iterator (Iter)
for (index, element) in list.iter().enumerate() {
assert_eq!(index, *element);
}
// You can get the current iteration index
// iter.index() == iter.len() means iteration ended (iter.next() == None)
let mut iter = &mut list.iter();
assert_eq!(iter.index(), 0);
assert_eq!(iter.next(), Some(&0));
assert_eq!(iter.index(), 1);
// List can also be cleared (but current iterators are not affected)
list.clear();
assert_eq!(iter.len(), 3);
assert_eq!(list.len(), 0);
assert_eq!(list.iter().len(), 0);
assert_eq!((&mut list.iter()).next(), None);
println!("Single thread example ended without errors");
}
use voluntary_servitude::vs;
use std::{sync::Arc, thread::spawn};
const CONSUMERS: usize = 8;
const PRODUCERS: usize = 4;
const ELEMENTS: usize = 10_000_000;
fn main() {
let list = Arc::new(vs![]);
let mut handlers = vec![];
// Creates producer threads to insert 10k elements
for _ in 0..PRODUCERS {
let l = Arc::clone(&list);
handlers.push(spawn(move || {
let _ = (0..ELEMENTS).map(|i| l.append(i)).count();
}));
}
// Creates consumer threads to print number of elements
// Until all of them are inserted
for _ in 0..CONSUMERS {
const TOTAL: usize = PRODUCERS * ELEMENTS;
let consumer = Arc::clone(&list);
handlers.push(spawn(move || loop {
let count = consumer.iter().count();
println!("{} elements", count);
if count >= TOTAL { break };
}));
}
// Join threads
for handler in handlers.into_iter() {
handler.join().expect("Failed to join thread");
}
println!("Multi-thread example ended without errors");
}