pub struct Memo<T>where
T: 'static,{ /* private fields */ }
Expand description
Memos are the result of computing a value from use_memo
.
You may have noticed that this struct doesn’t have many methods. Most methods for Memo
are defined on the Readable
and Writable
traits.
§Reading a Memo
You can use the methods on the Readable
trait to read a memo:
fn app() -> Element {
let mut count = use_signal(|| 0);
// The memo will rerun any time we write to the count signal
let halved = use_memo(move || count() / 2);
rsx! {
// When we read the value of memo, the current component will subscribe to the result of the memo. It will only rerun when the result of the memo changes.
"{halved}"
button {
onclick: move |_| {
count += 1;
},
"Increment"
}
}
}
Memo also includes helper methods just like Signal
s to make it easier to use. Calling a memo like a function will clone the inner value:
fn app() -> Element {
let mut count = use_signal(|| 0);
// The memo will rerun any time we write to the count signal
let halved = use_memo(move || count() / 2);
// This will rerun any time the halved value changes
let doubled = use_memo(move || 2 * halved());
rsx! {
"{doubled}"
button {
onclick: move |_| {
count += 1;
},
"Increment"
}
}
}
For a full list of all the helpers available, check out the Readable
, [ReadableVecExt
], and [ReadableOptionExt
] traits.
§Memos with Async
Because Memos check borrows at runtime, you need to be careful when reading memos inside of async code. If you hold a read of a memo over an await point, that read may still be open when the memo reruns which will cause a panic:
async fn double_me_async(value: &u32) -> u32 {
sleep(100).await;
*value * 2
}
let mut signal = use_signal(|| 0);
let halved = use_memo(move || signal() / 2);
let doubled = use_resource(move || async move {
// Don't hold reads over await points
let halved = halved.read();
// While the future is waiting for the async work to finish, the read will be open
double_me_async(&halved).await
});
rsx!{
"{doubled:?}"
button {
onclick: move |_| {
// When you write to signal, it will cause the memo to rerun which may panic because you are holding a read of the memo over an await point
signal += 1;
},
"Increment"
}
};
Instead of holding a read over an await point, you can clone whatever values you need out of your memo:
async fn double_me_async(value: u32) -> u32 {
sleep(100).await;
value * 2
}
let mut signal = use_signal(|| 0);
let halved = use_memo(move || signal() / 2);
let doubled = use_resource(move || async move {
// Calling the memo will clone the inner value
let halved = halved();
double_me_async(halved).await;
});
rsx!{
"{doubled:?}"
button {
onclick: move |_| {
signal += 1;
},
"Increment"
}
};
§Memo lifecycle
Memos are implemented with generational-box which makes all values Copy even if the inner value is not Copy.
This is incredibly convenient for UI development, but it does come with some tradeoffs. The lifetime of the memo is tied to the lifetime of the component it was created in. If you drop the component that created the memo, the memo will be dropped as well. You might run into this if you try to pass a memo from a child component to a parent component and drop the child component. To avoid this you can create your memo higher up in your component tree, or use global memos.
TLDR Don’t pass memos up in the component tree. It will cause issues:
fn MyComponent() -> Element {
let child_signal = use_signal(|| None);
rsx! {
IncrementButton {
child_signal
}
}
}
#[component]
fn IncrementButton(mut child_signal: Signal<Option<Memo<i32>>>) -> Element {
let signal_owned_by_child = use_signal(|| 0);
let memo_owned_by_child = use_memo(move || signal_owned_by_child() * 2);
// Don't do this: it may cause issues if you drop the child component
child_signal.set(Some(memo_owned_by_child));
todo!()
}
Implementations§
source§impl<T> Memo<T>where
T: 'static,
impl<T> Memo<T>where
T: 'static,
sourcepub fn new_with_location(
f: impl FnMut() -> T + 'static,
location: &'static Location<'static>,
) -> Memo<T>where
T: PartialEq,
pub fn new_with_location(
f: impl FnMut() -> T + 'static,
location: &'static Location<'static>,
) -> Memo<T>where
T: PartialEq,
Create a new memo with an explicit location
sourcepub const fn global(constructor: fn() -> T) -> Global<Memo<T>, T>where
T: PartialEq,
pub const fn global(constructor: fn() -> T) -> Global<Memo<T>, T>where
T: PartialEq,
Creates a new GlobalMemo
that can be used anywhere inside your dioxus app. This memo will automatically be created once per app the first time you use it.
§Example
static SIGNAL: GlobalSignal<i32> = Signal::global(|| 0);
// Create a new global memo that can be used anywhere in your app
static DOUBLED: GlobalMemo<i32> = Memo::global(|| SIGNAL() * 2);
fn App() -> Element {
rsx! {
button {
// When SIGNAL changes, the memo will update because the SIGNAL is read inside DOUBLED
onclick: move |_| *SIGNAL.write() += 1,
"{DOUBLED}"
}
}
}
Global memos are generally not recommended for use in libraries because it makes it more difficult to allow multiple instances of components you define in your library.
sourcepub fn origin_scope(&self) -> ScopeId
pub fn origin_scope(&self) -> ScopeId
Get the scope that the signal was created in.
sourcepub fn id(&self) -> GenerationalBoxId
pub fn id(&self) -> GenerationalBoxId
Get the id of the signal.
Trait Implementations§
source§impl<T> From<Memo<T>> for ReadOnlySignal<T>where
T: PartialEq,
impl<T> From<Memo<T>> for ReadOnlySignal<T>where
T: PartialEq,
source§fn from(val: Memo<T>) -> ReadOnlySignal<T>
fn from(val: Memo<T>) -> ReadOnlySignal<T>
source§impl<T> InitializeFromFunction<T> for Memo<T>where
T: PartialEq,
impl<T> InitializeFromFunction<T> for Memo<T>where
T: PartialEq,
source§fn initialize_from_function(f: fn() -> T) -> Memo<T>
fn initialize_from_function(f: fn() -> T) -> Memo<T>
source§impl<T> IntoAttributeValue for Memo<T>
impl<T> IntoAttributeValue for Memo<T>
source§fn into_value(self) -> AttributeValue
fn into_value(self) -> AttributeValue
source§impl<T> IntoDynNode for Memo<T>
impl<T> IntoDynNode for Memo<T>
source§fn into_dyn_node(self) -> DynamicNode
fn into_dyn_node(self) -> DynamicNode
source§impl<T> Readable for Memo<T>where
T: PartialEq,
impl<T> Readable for Memo<T>where
T: PartialEq,
source§fn try_peek_unchecked(
&self,
) -> Result<<<Memo<T> as Readable>::Storage as AnyStorage>::Ref<'static, <Memo<T> as Readable>::Target>, BorrowError>
fn try_peek_unchecked( &self, ) -> Result<<<Memo<T> as Readable>::Storage as AnyStorage>::Ref<'static, <Memo<T> as Readable>::Target>, BorrowError>
Get the current value of the signal. Unlike read, this will not subscribe the current scope to the signal which can cause parts of your UI to not update.
If the signal has been dropped, this will panic.
source§type Storage = UnsyncStorage
type Storage = UnsyncStorage
source§fn try_read_unchecked(
&self,
) -> Result<<<Memo<T> as Readable>::Storage as AnyStorage>::Ref<'static, <Memo<T> as Readable>::Target>, BorrowError>
fn try_read_unchecked( &self, ) -> Result<<<Memo<T> as Readable>::Storage as AnyStorage>::Ref<'static, <Memo<T> as Readable>::Target>, BorrowError>
source§fn map<O>(
self,
f: impl Fn(&Self::Target) -> &O + 'static,
) -> MappedSignal<O, Self::Storage>
fn map<O>( self, f: impl Fn(&Self::Target) -> &O + 'static, ) -> MappedSignal<O, Self::Storage>
source§fn read(&self) -> <Self::Storage as AnyStorage>::Ref<'_, Self::Target>
fn read(&self) -> <Self::Storage as AnyStorage>::Ref<'_, Self::Target>
source§fn try_read(
&self,
) -> Result<<Self::Storage as AnyStorage>::Ref<'_, Self::Target>, BorrowError>
fn try_read( &self, ) -> Result<<Self::Storage as AnyStorage>::Ref<'_, Self::Target>, BorrowError>
source§fn read_unchecked(
&self,
) -> <Self::Storage as AnyStorage>::Ref<'static, Self::Target>
fn read_unchecked( &self, ) -> <Self::Storage as AnyStorage>::Ref<'static, Self::Target>
source§fn peek(&self) -> <Self::Storage as AnyStorage>::Ref<'_, Self::Target>
fn peek(&self) -> <Self::Storage as AnyStorage>::Ref<'_, Self::Target>
source§fn try_peek(
&self,
) -> Result<<Self::Storage as AnyStorage>::Ref<'_, Self::Target>, BorrowError>
fn try_peek( &self, ) -> Result<<Self::Storage as AnyStorage>::Ref<'_, Self::Target>, BorrowError>
source§fn peek_unchecked(
&self,
) -> <Self::Storage as AnyStorage>::Ref<'static, Self::Target>
fn peek_unchecked( &self, ) -> <Self::Storage as AnyStorage>::Ref<'static, Self::Target>
source§fn with<O>(&self, f: impl FnOnce(&Self::Target) -> O) -> O
fn with<O>(&self, f: impl FnOnce(&Self::Target) -> O) -> O
impl<T> Copy for Memo<T>where
T: 'static,
Auto Trait Implementations§
impl<T> Freeze for Memo<T>
impl<T> !RefUnwindSafe for Memo<T>
impl<T> !Send for Memo<T>
impl<T> !Sync for Memo<T>
impl<T> Unpin for Memo<T>where
T: Unpin,
impl<T> !UnwindSafe for Memo<T>
Blanket Implementations§
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> CloneToUninit for Twhere
T: Clone,
impl<T> CloneToUninit for Twhere
T: Clone,
source§unsafe fn clone_to_uninit(&self, dst: *mut T)
unsafe fn clone_to_uninit(&self, dst: *mut T)
clone_to_uninit
)