Struct GILOnceCell
pub struct GILOnceCell<T> { /* private fields */ }
dep_pyo3
and std
only.Expand description
A write-once primitive similar to std::sync::OnceLock<T>
.
Unlike OnceLock<T>
which blocks threads to achieve thread safety, GilOnceCell<T>
allows calls to get_or_init
and
get_or_try_init
to race to create an initialized value.
(It is still guaranteed that only one thread will ever write to the cell.)
On Python versions that run with the Global Interpreter Lock (GIL), this helps to avoid deadlocks between initialization and the GIL. For an example of such a deadlock, see the FAQ section of the guide.
Note that because the GIL blocks concurrent execution, in practice the means that
get_or_init
and
get_or_try_init
may race if the initialization
function leads to the GIL being released and a thread context switch. This can
happen when importing or calling any Python code, as long as it releases the
GIL at some point. On free-threaded Python without any GIL, the race is
more likely since there is no GIL to prevent races. In the future, PyO3 may change
the semantics of GILOnceCell to behave more like the GIL build in the future.
§Re-entrant initialization
get_or_init
and
get_or_try_init
do not protect against infinite recursion
from reentrant initialization.
§Examples
The following example shows how to use GILOnceCell
to share a reference to a Python list
between threads:
use pyo3::sync::GILOnceCell;
use pyo3::prelude::*;
use pyo3::types::PyList;
static LIST_CELL: GILOnceCell<Py<PyList>> = GILOnceCell::new();
pub fn get_shared_list(py: Python<'_>) -> &Bound<'_, PyList> {
LIST_CELL
.get_or_init(py, || PyList::empty(py).unbind())
.bind(py)
}
Implementations§
§impl<T> GILOnceCell<T>
impl<T> GILOnceCell<T>
pub const fn new() -> GILOnceCell<T>
pub const fn new() -> GILOnceCell<T>
Create a GILOnceCell
which does not yet contain a value.
pub fn get(&self, _py: Python<'_>) -> Option<&T> ⓘ
pub fn get(&self, _py: Python<'_>) -> Option<&T> ⓘ
Get a reference to the contained value, or None
if the cell has not yet been written.
pub fn get_or_init<F>(&self, py: Python<'_>, f: F) -> &Twhere
F: FnOnce() -> T,
pub fn get_or_init<F>(&self, py: Python<'_>, f: F) -> &Twhere
F: FnOnce() -> T,
Get a reference to the contained value, initializing it if needed using the provided closure.
See the type-level documentation for detail on re-entrancy and concurrent initialization.
pub fn get_or_try_init<F, E>(&self, py: Python<'_>, f: F) -> Result<&T, E> ⓘ
pub fn get_or_try_init<F, E>(&self, py: Python<'_>, f: F) -> Result<&T, E> ⓘ
Like get_or_init
, but accepts a fallible initialization function. If it fails, the cell
is left uninitialized.
See the type-level documentation for detail on re-entrancy and concurrent initialization.
pub fn get_mut(&mut self) -> Option<&mut T> ⓘ
pub fn get_mut(&mut self) -> Option<&mut T> ⓘ
Get the contents of the cell mutably. This is only possible if the reference to the cell is unique.
pub fn set(&self, _py: Python<'_>, value: T) -> Result<(), T> ⓘ
pub fn set(&self, _py: Python<'_>, value: T) -> Result<(), T> ⓘ
Set the value in the cell.
If the cell has already been written, Err(value)
will be returned containing the new
value which was not written.
pub fn take(&mut self) -> Option<T> ⓘ
pub fn take(&mut self) -> Option<T> ⓘ
Takes the value out of the cell, moving it back to an uninitialized state.
Has no effect and returns None if the cell has not yet been written.
pub fn into_inner(self) -> Option<T> ⓘ
pub fn into_inner(self) -> Option<T> ⓘ
Consumes the cell, returning the wrapped value.
Returns None if the cell has not yet been written.
§impl<T> GILOnceCell<Py<T>>
impl<T> GILOnceCell<Py<T>>
pub fn clone_ref(&self, py: Python<'_>) -> GILOnceCell<Py<T>>
pub fn clone_ref(&self, py: Python<'_>) -> GILOnceCell<Py<T>>
Creates a new cell that contains a new Python reference to the same contained object.
Returns an uninitialized cell if self
has not yet been initialized.
§impl<T> GILOnceCell<Py<T>>where
T: PyTypeCheck,
impl<T> GILOnceCell<Py<T>>where
T: PyTypeCheck,
pub fn import<'py>(
&self,
py: Python<'py>,
module_name: &str,
attr_name: &str,
) -> Result<&Bound<'py, T>, PyErr> ⓘ
pub fn import<'py>( &self, py: Python<'py>, module_name: &str, attr_name: &str, ) -> Result<&Bound<'py, T>, PyErr> ⓘ
Get a reference to the contained Python type, initializing the cell if needed.
This is a shorthand method for get_or_init
which imports the type from Python on init.
§Example: Using GILOnceCell
to store a class in a static variable.
GILOnceCell
can be used to avoid importing a class multiple times:
#[pyfunction]
fn create_ordered_dict<'py>(py: Python<'py>, dict: Bound<'py, PyDict>) -> PyResult<Bound<'py, PyAny>> {
// Even if this function is called multiple times,
// the `OrderedDict` class will be imported only once.
static ORDERED_DICT: GILOnceCell<Py<PyType>> = GILOnceCell::new();
ORDERED_DICT
.import(py, "collections", "OrderedDict")?
.call1((dict,))
}
Trait Implementations§
§impl<T> Default for GILOnceCell<T>
impl<T> Default for GILOnceCell<T>
§fn default() -> GILOnceCell<T>
fn default() -> GILOnceCell<T>
impl<T> Send for GILOnceCell<T>where
T: Send,
impl<T> Sync for GILOnceCell<T>
Auto Trait Implementations§
impl<T> !Freeze for GILOnceCell<T>
impl<T> !RefUnwindSafe for GILOnceCell<T>
impl<T> Unpin for GILOnceCell<T>where
T: Unpin,
impl<T> UnwindSafe for GILOnceCell<T>where
T: UnwindSafe,
Blanket Implementations§
§impl<T> ArchivePointee for T
impl<T> ArchivePointee for T
§type ArchivedMetadata = ()
type ArchivedMetadata = ()
§fn pointer_metadata(
_: &<T as ArchivePointee>::ArchivedMetadata,
) -> <T as Pointee>::Metadata
fn pointer_metadata( _: &<T as ArchivePointee>::ArchivedMetadata, ) -> <T as Pointee>::Metadata
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
Source§impl<T> ByteSized for T
impl<T> ByteSized for T
Source§const BYTE_ALIGN: usize = _
const BYTE_ALIGN: usize = _
Source§fn byte_align(&self) -> usize ⓘ
fn byte_align(&self) -> usize ⓘ
Source§fn ptr_size_ratio(&self) -> [usize; 2]
fn ptr_size_ratio(&self) -> [usize; 2]
Source§impl<T, R> Chain<R> for Twhere
T: ?Sized,
impl<T, R> Chain<R> for Twhere
T: ?Sized,
Source§impl<T> ExtAny for T
impl<T> ExtAny for T
Source§fn as_any_mut(&mut self) -> &mut dyn Anywhere
Self: Sized,
fn as_any_mut(&mut self) -> &mut dyn Anywhere
Self: Sized,
Source§impl<T> ExtMem for Twhere
T: ?Sized,
impl<T> ExtMem for Twhere
T: ?Sized,
Source§const NEEDS_DROP: bool = _
const NEEDS_DROP: bool = _
Source§fn mem_align_of_val(&self) -> usize ⓘ
fn mem_align_of_val(&self) -> usize ⓘ
Source§fn mem_size_of_val(&self) -> usize ⓘ
fn mem_size_of_val(&self) -> usize ⓘ
Source§fn mem_needs_drop(&self) -> bool
fn mem_needs_drop(&self) -> bool
true
if dropping values of this type matters. Read moreSource§fn mem_forget(self)where
Self: Sized,
fn mem_forget(self)where
Self: Sized,
self
without running its destructor. Read moreSource§fn mem_replace(&mut self, other: Self) -> Selfwhere
Self: Sized,
fn mem_replace(&mut self, other: Self) -> Selfwhere
Self: Sized,
Source§unsafe fn mem_zeroed<T>() -> T
unsafe fn mem_zeroed<T>() -> T
unsafe_layout
only.T
represented by the all-zero byte-pattern. Read moreSource§unsafe fn mem_transmute_copy<Src, Dst>(src: &Src) -> Dst
unsafe fn mem_transmute_copy<Src, Dst>(src: &Src) -> Dst
unsafe_layout
only.T
represented by the all-zero byte-pattern. Read moreSource§fn mem_as_bytes(&self) -> &[u8] ⓘ
fn mem_as_bytes(&self) -> &[u8] ⓘ
unsafe_slice
only.§impl<S> FromSample<S> for S
impl<S> FromSample<S> for S
fn from_sample_(s: S) -> S
Source§impl<T> Hook for T
impl<T> Hook for T
§impl<T> Instrument for T
impl<T> Instrument for T
§fn instrument(self, span: Span) -> Instrumented<Self> ⓘ
fn instrument(self, span: Span) -> Instrumented<Self> ⓘ
§fn in_current_span(self) -> Instrumented<Self> ⓘ
fn in_current_span(self) -> Instrumented<Self> ⓘ
Source§impl<T> IntoEither for T
impl<T> IntoEither for T
Source§fn into_either(self, into_left: bool) -> Either<Self, Self> ⓘ
fn into_either(self, into_left: bool) -> Either<Self, Self> ⓘ
self
into a Left
variant of Either<Self, Self>
if into_left
is true
.
Converts self
into a Right
variant of Either<Self, Self>
otherwise. Read moreSource§fn into_either_with<F>(self, into_left: F) -> Either<Self, Self> ⓘ
fn into_either_with<F>(self, into_left: F) -> Either<Self, Self> ⓘ
self
into a Left
variant of Either<Self, Self>
if into_left(&self)
returns true
.
Converts self
into a Right
variant of Either<Self, Self>
otherwise. Read more§impl<F, T> IntoSample<T> for Fwhere
T: FromSample<F>,
impl<F, T> IntoSample<T> for Fwhere
T: FromSample<F>,
fn into_sample(self) -> T
§impl<T> LayoutRaw for T
impl<T> LayoutRaw for T
§fn layout_raw(_: <T as Pointee>::Metadata) -> Result<Layout, LayoutError> ⓘ
fn layout_raw(_: <T as Pointee>::Metadata) -> Result<Layout, LayoutError> ⓘ
§impl<T, N1, N2> Niching<NichedOption<T, N1>> for N2
impl<T, N1, N2> Niching<NichedOption<T, N1>> for N2
§unsafe fn is_niched(niched: *const NichedOption<T, N1>) -> bool
unsafe fn is_niched(niched: *const NichedOption<T, N1>) -> bool
§fn resolve_niched(out: Place<NichedOption<T, N1>>)
fn resolve_niched(out: Place<NichedOption<T, N1>>)
out
indicating that a T
is niched.