Struct otter_nodejs_tests::thread::LocalKey  
1.0.0 · source · [−]pub struct LocalKey<T> where
    T: 'static,  { /* private fields */ }Expand description
A thread local storage key which owns its contents.
This key uses the fastest possible implementation available to it for the
target platform. It is instantiated with the thread_local! macro and the
primary method is the with method.
The with method yields a reference to the contained value which cannot be
sent across threads or escape the given closure.
Initialization and Destruction
Initialization is dynamically performed on the first call to with
within a thread, and values that implement Drop get destructed when a
thread exits. Some caveats apply, which are explained below.
A LocalKey’s initializer cannot recursively depend on itself, and using
a LocalKey in this way will cause the initializer to infinitely recurse
on the first call to with.
Examples
use std::cell::RefCell;
use std::thread;
thread_local!(static FOO: RefCell<u32> = RefCell::new(1));
FOO.with(|f| {
    assert_eq!(*f.borrow(), 1);
    *f.borrow_mut() = 2;
});
// each thread starts out with the initial value of 1
let t = thread::spawn(move|| {
    FOO.with(|f| {
        assert_eq!(*f.borrow(), 1);
        *f.borrow_mut() = 3;
    });
});
// wait for the thread to complete and bail out on panic
t.join().unwrap();
// we retain our original value of 2 despite the child thread
FOO.with(|f| {
    assert_eq!(*f.borrow(), 2);
});Platform-specific behavior
Note that a “best effort” is made to ensure that destructors for types stored in thread local storage are run, but not all platforms can guarantee that destructors will be run for all types in thread local storage. For example, there are a number of known caveats where destructors are not run:
- On Unix systems when pthread-based TLS is being used, destructors will not be run for TLS values on the main thread when it exits. Note that the application will exit immediately after the main thread exits as well.
- On all platforms it’s possible for TLS to re-initialize other TLS slots during destruction. Some platforms ensure that this cannot happen infinitely by preventing re-initialization of any slot that has been destroyed, but not all platforms have this guard. Those platforms that do not guard typically have a synthetic limit after which point no more destructors are run.
- When the process exits on Windows systems, TLS destructors may only be run on the thread that causes the process to exit. This is because the other threads may be forcibly terminated.
Synchronization in thread-local destructors
On Windows, synchronization operations (such as JoinHandle::join) in
thread local destructors are prone to deadlocks and so should be avoided.
This is because the loader lock is held while a destructor is run. The
lock is acquired whenever a thread starts or exits or when a DLL is loaded
or unloaded. Therefore these events are blocked for as long as a thread
local destructor is running.
Implementations
sourceimpl<T> LocalKey<T> where
    T: 'static, 
 
impl<T> LocalKey<T> where
    T: 'static, 
sourcepub fn with<F, R>(&'static self, f: F) -> R where
    F: FnOnce(&T) -> R, 
 
pub fn with<F, R>(&'static self, f: F) -> R where
    F: FnOnce(&T) -> R, 
Acquires a reference to the value in this TLS key.
This will lazily initialize the value if this thread has not referenced this key yet.
Panics
This function will panic!() if the key currently has its
destructor running, and it may panic if the destructor has
previously been run for this thread.
1.26.0 · sourcepub fn try_with<F, R>(&'static self, f: F) -> Result<R, AccessError> where
    F: FnOnce(&T) -> R, 
 
pub fn try_with<F, R>(&'static self, f: F) -> Result<R, AccessError> where
    F: FnOnce(&T) -> R, 
Acquires a reference to the value in this TLS key.
This will lazily initialize the value if this thread has not referenced
this key yet. If the key has been destroyed (which may happen if this is called
in a destructor), this function will return an AccessError.
Panics
This function will still panic!() if the key is uninitialized and the
key’s initializer panics.
sourceimpl<T> LocalKey<Cell<T>> where
    T: 'static, 
 
impl<T> LocalKey<Cell<T>> where
    T: 'static, 
sourcepub fn set(&'static self, value: T)
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn set(&'static self, value: T)
local_key_cell_methods)Sets or initializes the contained value.
Unlike the other methods, this will not run the lazy initializer of the thread local. Instead, it will be directly initialized with the given value if it wasn’t initialized yet.
Panics
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Examples
#![feature(local_key_cell_methods)]
use std::cell::Cell;
thread_local! {
    static X: Cell<i32> = panic!("!");
}
// Calling X.get() here would result in a panic.
X.set(123); // But X.set() is fine, as it skips the initializer above.
assert_eq!(X.get(), 123);sourcepub fn get(&'static self) -> T where
    T: Copy, 
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn get(&'static self) -> T where
    T: Copy, 
local_key_cell_methods)Returns a copy of the contained value.
This will lazily initialize the value if this thread has not referenced this key yet.
Panics
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Examples
#![feature(local_key_cell_methods)]
use std::cell::Cell;
thread_local! {
    static X: Cell<i32> = Cell::new(1);
}
assert_eq!(X.get(), 1);sourcepub fn take(&'static self) -> T where
    T: Default, 
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn take(&'static self) -> T where
    T: Default, 
local_key_cell_methods)Takes the contained value, leaving Default::default() in its place.
This will lazily initialize the value if this thread has not referenced this key yet.
Panics
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Examples
#![feature(local_key_cell_methods)]
use std::cell::Cell;
thread_local! {
    static X: Cell<Option<i32>> = Cell::new(Some(1));
}
assert_eq!(X.take(), Some(1));
assert_eq!(X.take(), None);sourcepub fn replace(&'static self, value: T) -> T
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn replace(&'static self, value: T) -> T
local_key_cell_methods)Replaces the contained value, returning the old value.
This will lazily initialize the value if this thread has not referenced this key yet.
Panics
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Examples
#![feature(local_key_cell_methods)]
use std::cell::Cell;
thread_local! {
    static X: Cell<i32> = Cell::new(1);
}
assert_eq!(X.replace(2), 1);
assert_eq!(X.replace(3), 2);sourceimpl<T> LocalKey<RefCell<T>> where
    T: 'static, 
 
impl<T> LocalKey<RefCell<T>> where
    T: 'static, 
sourcepub fn with_borrow<F, R>(&'static self, f: F) -> R where
    F: FnOnce(&T) -> R, 
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn with_borrow<F, R>(&'static self, f: F) -> R where
    F: FnOnce(&T) -> R, 
local_key_cell_methods)Acquires a reference to the contained value.
This will lazily initialize the value if this thread has not referenced this key yet.
Panics
Panics if the value is currently mutably borrowed.
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Example
#![feature(local_key_cell_methods)]
use std::cell::RefCell;
thread_local! {
    static X: RefCell<Vec<i32>> = RefCell::new(Vec::new());
}
X.with_borrow(|v| assert!(v.is_empty()));sourcepub fn with_borrow_mut<F, R>(&'static self, f: F) -> R where
    F: FnOnce(&mut T) -> R, 
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn with_borrow_mut<F, R>(&'static self, f: F) -> R where
    F: FnOnce(&mut T) -> R, 
local_key_cell_methods)Acquires a mutable reference to the contained value.
This will lazily initialize the value if this thread has not referenced this key yet.
Panics
Panics if the value is currently borrowed.
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Example
#![feature(local_key_cell_methods)]
use std::cell::RefCell;
thread_local! {
    static X: RefCell<Vec<i32>> = RefCell::new(Vec::new());
}
X.with_borrow_mut(|v| v.push(1));
X.with_borrow(|v| assert_eq!(*v, vec![1]));sourcepub fn set(&'static self, value: T)
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn set(&'static self, value: T)
local_key_cell_methods)Sets or initializes the contained value.
Unlike the other methods, this will not run the lazy initializer of the thread local. Instead, it will be directly initialized with the given value if it wasn’t initialized yet.
Panics
Panics if the value is currently borrowed.
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Examples
#![feature(local_key_cell_methods)]
use std::cell::RefCell;
thread_local! {
    static X: RefCell<Vec<i32>> = panic!("!");
}
// Calling X.with() here would result in a panic.
X.set(vec![1, 2, 3]); // But X.set() is fine, as it skips the initializer above.
X.with_borrow(|v| assert_eq!(*v, vec![1, 2, 3]));sourcepub fn take(&'static self) -> T where
    T: Default, 
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn take(&'static self) -> T where
    T: Default, 
local_key_cell_methods)Takes the contained value, leaving Default::default() in its place.
This will lazily initialize the value if this thread has not referenced this key yet.
Panics
Panics if the value is currently borrowed.
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Examples
#![feature(local_key_cell_methods)]
use std::cell::RefCell;
thread_local! {
    static X: RefCell<Vec<i32>> = RefCell::new(Vec::new());
}
X.with_borrow_mut(|v| v.push(1));
let a = X.take();
assert_eq!(a, vec![1]);
X.with_borrow(|v| assert!(v.is_empty()));sourcepub fn replace(&'static self, value: T) -> T
 🔬 This is a nightly-only experimental API. (local_key_cell_methods)
pub fn replace(&'static self, value: T) -> T
local_key_cell_methods)Replaces the contained value, returning the old value.
Panics
Panics if the value is currently borrowed.
Panics if the key currently has its destructor running, and it may panic if the destructor has previously been run for this thread.
Examples
#![feature(local_key_cell_methods)]
use std::cell::RefCell;
thread_local! {
    static X: RefCell<Vec<i32>> = RefCell::new(Vec::new());
}
let prev = X.replace(vec![1, 2, 3]);
assert!(prev.is_empty());
X.with_borrow(|v| assert_eq!(*v, vec![1, 2, 3]));Trait Implementations
Auto Trait Implementations
impl<T> RefUnwindSafe for LocalKey<T>
impl<T> Send for LocalKey<T>
impl<T> Sync for LocalKey<T>
impl<T> Unpin for LocalKey<T>
impl<T> UnwindSafe for LocalKey<T>
Blanket Implementations
sourceimpl<T> BorrowMut<T> for T where
    T: ?Sized, 
 
impl<T> BorrowMut<T> for T where
    T: ?Sized, 
const: unstable · sourcefn borrow_mut(&mut self) -> &mut T
 
fn borrow_mut(&mut self) -> &mut T
Mutably borrows from an owned value. Read more
impl<T> Downcast for T where
    T: Any, 
impl<T> Downcast for T where
    T: Any, 
fn into_any(self: Box<T, Global>) -> Box<dyn Any + 'static, Global>ⓘNotable traits for Box<R, Global>impl<R> Read for Box<R, Global> where
    R: Read + ?Sized, impl<W> Write for Box<W, Global> where
    W: Write + ?Sized, impl<F, A> Future for Box<F, A> where
    F: Future + Unpin + ?Sized,
    A: Allocator + 'static,     type Output = <F as Future>::Output;impl<I, A> Iterator for Box<I, A> where
    I: Iterator + ?Sized,
    A: Allocator,     type Item = <I as Iterator>::Item;
fn into_any(self: Box<T, Global>) -> Box<dyn Any + 'static, Global>ⓘNotable traits for Box<R, Global>impl<R> Read for Box<R, Global> where
    R: Read + ?Sized, impl<W> Write for Box<W, Global> where
    W: Write + ?Sized, impl<F, A> Future for Box<F, A> where
    F: Future + Unpin + ?Sized,
    A: Allocator + 'static,     type Output = <F as Future>::Output;impl<I, A> Iterator for Box<I, A> where
    I: Iterator + ?Sized,
    A: Allocator,     type Item = <I as Iterator>::Item;
R: Read + ?Sized, impl<W> Write for Box<W, Global> where
W: Write + ?Sized, impl<F, A> Future for Box<F, A> where
F: Future + Unpin + ?Sized,
A: Allocator + 'static, type Output = <F as Future>::Output;impl<I, A> Iterator for Box<I, A> where
I: Iterator + ?Sized,
A: Allocator, type Item = <I as Iterator>::Item;
Convert Box<dyn Trait> (where Trait: Downcast) to Box<dyn Any>. Box<dyn Any> can
then be further downcast into Box<ConcreteType> where ConcreteType implements Trait. Read more
fn into_any_rc(self: Rc<T>) -> Rc<dyn Any + 'static>
fn into_any_rc(self: Rc<T>) -> Rc<dyn Any + 'static>
Convert Rc<Trait> (where Trait: Downcast) to Rc<Any>. Rc<Any> can then be
further downcast into Rc<ConcreteType> where ConcreteType implements Trait. Read more
fn as_any(&self) -> &(dyn Any + 'static)
fn as_any(&self) -> &(dyn Any + 'static)
Convert &Trait (where Trait: Downcast) to &Any. This is needed since Rust cannot
generate &Any’s vtable from &Trait’s. Read more
fn as_any_mut(&mut self) -> &mut (dyn Any + 'static)
fn as_any_mut(&mut self) -> &mut (dyn Any + 'static)
Convert &mut Trait (where Trait: Downcast) to &Any. This is needed since Rust cannot
generate &mut Any’s vtable from &mut Trait’s. Read more
impl<A> DynCastExt for A
impl<A> DynCastExt for A
fn dyn_cast<T>(
    self
) -> Result<<A as DynCastExtHelper<T>>::Target, <A as DynCastExtHelper<T>>::Source> where
    A: DynCastExtHelper<T>,
    T: ?Sized, 
fn dyn_cast<T>(
    self
) -> Result<<A as DynCastExtHelper<T>>::Target, <A as DynCastExtHelper<T>>::Source> where
    A: DynCastExtHelper<T>,
    T: ?Sized, 
Use this to cast from one trait object type to another. Read more
fn dyn_upcast<T>(self) -> <A as DynCastExtAdvHelper<T, T>>::Target where
    A: DynCastExtAdvHelper<T, T, Source = <A as DynCastExtAdvHelper<T, T>>::Target>,
    T: ?Sized, 
fn dyn_upcast<T>(self) -> <A as DynCastExtAdvHelper<T, T>>::Target where
    A: DynCastExtAdvHelper<T, T, Source = <A as DynCastExtAdvHelper<T, T>>::Target>,
    T: ?Sized, 
Use this to upcast a trait to one of its supertraits. Read more
fn dyn_cast_adv<F, T>(
    self
) -> Result<<A as DynCastExtAdvHelper<F, T>>::Target, <A as DynCastExtAdvHelper<F, T>>::Source> where
    A: DynCastExtAdvHelper<F, T>,
    F: ?Sized,
    T: ?Sized, 
fn dyn_cast_adv<F, T>(
    self
) -> Result<<A as DynCastExtAdvHelper<F, T>>::Target, <A as DynCastExtAdvHelper<F, T>>::Source> where
    A: DynCastExtAdvHelper<F, T>,
    F: ?Sized,
    T: ?Sized, 
fn dyn_cast_with_config<C>(
    self
) -> Result<<A as DynCastExtAdvHelper<<C as DynCastConfig>::Source, <C as DynCastConfig>::Target>>::Target, <A as DynCastExtAdvHelper<<C as DynCastConfig>::Source, <C as DynCastConfig>::Target>>::Source> where
    C: DynCastConfig,
    A: DynCastExtAdvHelper<<C as DynCastConfig>::Source, <C as DynCastConfig>::Target>, 
fn dyn_cast_with_config<C>(
    self
) -> Result<<A as DynCastExtAdvHelper<<C as DynCastConfig>::Source, <C as DynCastConfig>::Target>>::Target, <A as DynCastExtAdvHelper<<C as DynCastConfig>::Source, <C as DynCastConfig>::Target>>::Source> where
    C: DynCastConfig,
    A: DynCastExtAdvHelper<<C as DynCastConfig>::Source, <C as DynCastConfig>::Target>, 
Use this to cast from one trait object type to another. With this method the type parameter is a config type that uniquely specifies which cast should be preformed. Read more
sourceimpl<T> Instrument for T
 
impl<T> Instrument for T
sourcefn instrument(self, span: Span) -> Instrumented<Self>ⓘNotable traits for Instrumented<T>impl<T> Future for Instrumented<T> where
    T: Future,     type Output = <T as Future>::Output;
 
fn instrument(self, span: Span) -> Instrumented<Self>ⓘNotable traits for Instrumented<T>impl<T> Future for Instrumented<T> where
    T: Future,     type Output = <T as Future>::Output;
T: Future, type Output = <T as Future>::Output;
sourcefn in_current_span(self) -> Instrumented<Self>ⓘNotable traits for Instrumented<T>impl<T> Future for Instrumented<T> where
    T: Future,     type Output = <T as Future>::Output;
 
fn in_current_span(self) -> Instrumented<Self>ⓘNotable traits for Instrumented<T>impl<T> Future for Instrumented<T> where
    T: Future,     type Output = <T as Future>::Output;
T: Future, type Output = <T as Future>::Output;
impl<V, T> VZip<V> for T where
    V: MultiLane<T>, 
impl<V, T> VZip<V> for T where
    V: MultiLane<T>, 
fn vzip(self) -> V
sourceimpl<T> WithSubscriber for T
 
impl<T> WithSubscriber for T
sourcefn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>ⓘNotable traits for WithDispatch<T>impl<T> Future for WithDispatch<T> where
    T: Future,     type Output = <T as Future>::Output; where
    S: Into<Dispatch>, 
 
fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>ⓘNotable traits for WithDispatch<T>impl<T> Future for WithDispatch<T> where
    T: Future,     type Output = <T as Future>::Output; where
    S: Into<Dispatch>, 
T: Future, type Output = <T as Future>::Output;
Attaches the provided Subscriber to this type, returning a
WithDispatch wrapper. Read more
sourcefn with_current_subscriber(self) -> WithDispatch<Self>ⓘNotable traits for WithDispatch<T>impl<T> Future for WithDispatch<T> where
    T: Future,     type Output = <T as Future>::Output;
 
fn with_current_subscriber(self) -> WithDispatch<Self>ⓘNotable traits for WithDispatch<T>impl<T> Future for WithDispatch<T> where
    T: Future,     type Output = <T as Future>::Output;
T: Future, type Output = <T as Future>::Output;
Attaches the current default Subscriber to this type, returning a
WithDispatch wrapper. Read more