| Crates.io | lazyinit |
| lib.rs | lazyinit |
| version | 0.2.2 |
| created_at | 2024-07-17 16:27:57.318246+00 |
| updated_at | 2025-06-10 17:38:22.729226+00 |
| description | Initialize a static value lazily. |
| homepage | https://github.com/arceos-org/arceos |
| repository | https://github.com/arceos-org/lazyinit |
| max_upload_size | |
| id | 1306338 |
| size | 13,736 |
Initialize a static value lazily.
The crate provides a type for initializing static values lazily in a thread-safe manner.
Unlike compile-time initialization or macro-based solutions like lazy_static, this type allows runtime initialization with arbitrary logic while guaranteeing that initialization occurs exactly once across all threads.
The core abstraction is a struct that wraps a value and manages its initialization state through atomic operations. The value remains uninitialized until the first call to init_once or call_once, at which point it becomes permanently initialized and accessible.
Deref and DerefMut for transparent access after initializationuse lazyinit::LazyInit;
static VALUE: LazyInit<u32> = LazyInit::new();
assert!(!VALUE.is_inited());
// println!("{}", *VALUE); // panic: use uninitialized value
assert_eq!(VALUE.get(), None);
VALUE.init_once(233);
// VALUE.init_once(666); // panic: already initialized
assert!(VALUE.is_inited());
assert_eq!(*VALUE, 233);
assert_eq!(VALUE.get(), Some(&233));
Only one of the multiple initializations can succeed:
use lazyinit::LazyInit;
use std::time::Duration;
const N: usize = 16;
static VALUE: LazyInit<usize> = LazyInit::new();
let threads = (0..N)
.map(|i| {
std::thread::spawn(move || {
std::thread::sleep(Duration::from_millis(10));
VALUE.call_once(|| i)
})
})
.collect::<Vec<_>>();
let mut ok = 0;
for (i, thread) in threads.into_iter().enumerate() {
if thread.join().unwrap().is_some() {
ok += 1;
assert_eq!(*VALUE, i);
}
}
assert_eq!(ok, 1);