Crates.io | atomic-interval |
lib.rs | atomic-interval |
version | 0.1.4 |
source | src |
created_at | 2021-11-15 22:44:33.374934 |
updated_at | 2024-02-15 19:12:14.068813 |
description | A tiny implementation of an atomic timer. |
homepage | |
repository | https://github.com/BiagioFesta/atomic-interval |
max_upload_size | |
id | 482389 |
size | 37,869 |
A very tiny library implementing a lock-free atomic timer.
In the following example, we have a concurrent scenario where multiple threads want to push a data sample towards a single and common entity (e.g., a static function, a multi-referenced object, and so on).
The entity wants to provide a "limiter" mechanism, where it actually processes a sample with a limited frequency.
Therefore, even if concurrently threads push with a higher frequency, only one sample (coming from one thread) for each period can be actually processed (i.e., printed).
AtomicInterval
can be used without an additional sync mechanism, as it can already be safely shared across threads.
use atomic_interval::AtomicIntervalLight;
use once_cell::sync::OnceCell;
use std::thread;
use std::time::Duration;
const MAX_PERIOD_SAMPLING: Duration = Duration::from_secs(1);
fn push_sample(id_thread: usize, value: u8) {
// Note AtomicInterval can be used without additional
// sync wrapper (e.g., a `Mutex`) as it is atomic.
static LIMITER: OnceCell<AtomicIntervalLight> = OnceCell::new();
let limiter_init = || AtomicIntervalLight::new(MAX_PERIOD_SAMPLING);
// Only one threads can push a sample for each PERIOD.
// We limit the samples acquisition with a interval.
if LIMITER.get_or_init(limiter_init).is_ticked() {
println!("Thread '{}' pushed sample: '{}'", id_thread, value);
}
}
fn main() {
let num_threads = num_cpus::get();
(0..num_threads)
.map(|id_thread| {
thread::spawn(move || loop {
let sample = rand::random();
// Multiple threads concurrently try to push a sample.
push_sample(id_thread, sample);
thread::sleep(Duration::from_millis(1));
})
})
.collect::<Vec<_>>()
.into_iter()
.for_each(|join_handle| join_handle.join().unwrap());
}