[][src]Enum maybe_owned::MaybeOwned

pub enum MaybeOwned<'a, T: 'a> {
    Owned(T),
    Borrowed(&'a T),
}

This type provides a way to store data to which you either have a reference to or which you do own.

It provides From<T>, From<&'a T> implementations and, in difference to Cow does not require ToOwned to be implemented which makes it compatible with non cloneable data, as a draw back of this it does not know about ToOwned. As a consequence of it can't know that &str should be the borrowed version of String and not &String this is especially bad wrt. Box as the borrowed version of Box<T> would be &Box<T>.

While this crate has some drawbacks compared to Cow is has the benefit, that it works with Types which neither implement Clone nor ToOwned. Another benefit lies in the ability to write API functions which accept a generic parameter E: Into<MaybeOwned<'a, T>> as the API consumer can pass T, &'a T and MaybeOwned<'a, T> as argument, without requiring a explicit Cow::Owned or a split into two functions one accepting owed and the other borrowed values.

Alternatives

If you mainly have values implementing ToOwned like &str/String, Path/PathBuf or &[T]/Vec<T> using std::borrow::Cow might be preferable.

If you want to be able to treat &T, &mut T, Box<T> and Arc<T> the same consider using reffers::rbma::RBMA (through not all types/platforms are supported because as it relies on the fact that for many pointers the lowest two bits are 0, and stores the discriminant in them, nevertheless this is can only be used with 32bit-aligned data, e.g. using a &u8 might fail). RBMA also allows you to recover a &mut T if it was created from Box<T>, &mut T or a unique Arc.

Examples

struct PseudoBigData(u8);
fn pseudo_register_fn<'a, E>(_val: E) where E: Into<MaybeOwned<'a, PseudoBigData>> { }

let data = PseudoBigData(12);
let data2 = PseudoBigData(13);

pseudo_register_fn(&data);
pseudo_register_fn(&data);
pseudo_register_fn(data2);
pseudo_register_fn(MaybeOwned::Owned(PseudoBigData(111)));
#[repr(C)]
struct OpaqueFFI {
    ref1:  * const u8
    //we also might want to have PhantomData etc.
}

// does not work as it does not implement `ToOwned`
// let _ = Cow::Owned(OpaqueFFI { ref1: 0 as *const u8});

// ok, MaybeOwned can do this (but can't do &str<->String as tread of)
let _ = MaybeOwned::Owned(OpaqueFFI { ref1: 0 as *const u8 });
use std::collections::HashMap;

#[derive(Serialize, Deserialize)]
struct SerializedData<'a> {
    data: MaybeOwned<'a, HashMap<String, i32>>,
}

let mut map = HashMap::new();
map.insert("answer".to_owned(), 42);

// serializing can use borrowed data to avoid unnecessary copying
let bytes = serde_json::to_vec(&SerializedData { data: (&map).into() }).unwrap();

// deserializing creates owned data
let deserialized: SerializedData = serde_json::from_slice(&bytes).unwrap();
assert_eq!(deserialized.data["answer"], 42);

Transitive std::ops implementations

There are transitive implementations for most operator in std::ops.

A Op between a MaybeOwned<L> and MaybeOwned<R> is implemented if:

  • L impl the Op with R
  • L impl the Op with &R
  • &L impl the Op with R
  • &L impl the Op with &R
  • the Output of all aboves implementations is the same type

The Neg (- prefix) op is implemented for V if:

  • V impl Neg
  • &V impl Neg
  • both have the same Output

The Not (! prefix) op is implemented for V if:

  • V impl Not
  • &V impl Not
  • both have the same Output

Adding implementations for Ops which add a MaybeOwned to a non MaybeOwned value (like MaybeOwned<T> + T) requires far reaching specialization in rust and is therefore not done for now.

Variants

Owned(T)

owns T

Borrowed(&'a T)

has a reference to T

Implementations

impl<'_, T> MaybeOwned<'_, T>[src]

pub fn is_owned(&self) -> bool[src]

returns true if the data is owned else false

impl<'_, T: Clone> MaybeOwned<'_, T>[src]

pub fn into_owned(self) -> T[src]

Extracts the owned data.

If the data is borrowed it is cloned before being extracted.

impl<'_, T: Clone> MaybeOwned<'_, T>[src]

pub fn to_mut(&mut self) -> &mut T[src]

Acquires a mutable reference to owned data.

Clones data if it is not already owned.

Example

use maybe_owned::MaybeOwned;

#[derive(Clone, Debug, PartialEq, Eq)]
struct PseudoBigData(u8);

let data = PseudoBigData(12);

let mut maybe: MaybeOwned<PseudoBigData> = (&data).into();
assert_eq!(false, maybe.is_owned());

{
    let reference = maybe.to_mut();
    assert_eq!(&mut PseudoBigData(12), reference);
}
assert!(maybe.is_owned());

Trait Implementations

impl<'min, L, R, OUT: 'min> Add<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Add<R, Output = OUT> + Add<&'min R, Output = OUT>,
    &'min L: Add<R, Output = OUT> + Add<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the + operator.

impl<'min, L, R> AddAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + AddAssign<R> + AddAssign<&'min R>, 
[src]

impl<'_, T> AsRef<T> for MaybeOwned<'_, T>[src]

impl<'min, L, R, OUT: 'min> BitAnd<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: BitAnd<R, Output = OUT> + BitAnd<&'min R, Output = OUT>,
    &'min L: BitAnd<R, Output = OUT> + BitAnd<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the & operator.

impl<'min, L, R> BitAndAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + BitAndAssign<R> + BitAndAssign<&'min R>, 
[src]

impl<'min, L, R, OUT: 'min> BitOr<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: BitOr<R, Output = OUT> + BitOr<&'min R, Output = OUT>,
    &'min L: BitOr<R, Output = OUT> + BitOr<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the | operator.

impl<'min, L, R> BitOrAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + BitOrAssign<R> + BitOrAssign<&'min R>, 
[src]

impl<'min, L, R, OUT: 'min> BitXor<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: BitXor<R, Output = OUT> + BitXor<&'min R, Output = OUT>,
    &'min L: BitXor<R, Output = OUT> + BitXor<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the ^ operator.

impl<'min, L, R> BitXorAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + BitXorAssign<R> + BitXorAssign<&'min R>, 
[src]

impl<'_, T> Borrow<T> for MaybeOwned<'_, T>[src]

impl<'_, T: Clone> Clone for MaybeOwned<'_, T>[src]

impl<'a, T: Debug + 'a> Debug for MaybeOwned<'a, T>[src]

impl<'_, T: Default> Default for MaybeOwned<'_, T>[src]

impl<'_, T> Deref for MaybeOwned<'_, T>[src]

type Target = T

The resulting type after dereferencing.

impl<'a, T: Display> Display for MaybeOwned<'a, T>[src]

impl<'min, L, R, OUT: 'min> Div<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Div<R, Output = OUT> + Div<&'min R, Output = OUT>,
    &'min L: Div<R, Output = OUT> + Div<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the / operator.

impl<'min, L, R> DivAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + DivAssign<R> + DivAssign<&'min R>, 
[src]

impl<'a, T: Eq> Eq for MaybeOwned<'a, T>[src]

impl<'a, T> From<&'a T> for MaybeOwned<'a, T>[src]

impl<'a, T: ToOwned<Owned = T>> From<Cow<'a, T>> for MaybeOwned<'a, T>[src]

impl<'_, T> From<T> for MaybeOwned<'_, T>[src]

impl<'_, T: FromStr> FromStr for MaybeOwned<'_, T>[src]

type Err = T::Err

The associated error which can be returned from parsing.

impl<'_, T: Hash> Hash for MaybeOwned<'_, T>[src]

impl<'a, T: ToOwned<Owned = T>> Into<Cow<'a, T>> for MaybeOwned<'a, T>[src]

impl<'min, L, R, OUT: 'min> Mul<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Mul<R, Output = OUT> + Mul<&'min R, Output = OUT>,
    &'min L: Mul<R, Output = OUT> + Mul<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the * operator.

impl<'min, L, R> MulAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + MulAssign<R> + MulAssign<&'min R>, 
[src]

impl<'l, V, OUT> Neg for MaybeOwned<'l, V> where
    V: Neg<Output = OUT>,
    &'l V: Neg<Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the - operator.

impl<'l, V, OUT> Not for MaybeOwned<'l, V> where
    V: Not<Output = OUT>,
    &'l V: Not<Output = OUT>, 
[src]

type Output = V::Output

The resulting type after applying the ! operator.

impl<'_, T: Ord> Ord for MaybeOwned<'_, T>[src]

impl<'b, '_, A: PartialEq<B>, B> PartialEq<MaybeOwned<'b, B>> for MaybeOwned<'_, A>[src]

impl<'_, T: PartialOrd> PartialOrd<MaybeOwned<'_, T>> for MaybeOwned<'_, T>[src]

impl<'min, L, R, OUT: 'min> Shl<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Shl<R, Output = OUT> + Shl<&'min R, Output = OUT>,
    &'min L: Shl<R, Output = OUT> + Shl<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the << operator.

impl<'min, L, R> ShlAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + ShlAssign<R> + ShlAssign<&'min R>, 
[src]

impl<'min, L, R, OUT: 'min> Shr<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Shr<R, Output = OUT> + Shr<&'min R, Output = OUT>,
    &'min L: Shr<R, Output = OUT> + Shr<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the >> operator.

impl<'min, L, R> ShrAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + ShrAssign<R> + ShrAssign<&'min R>, 
[src]

impl<'min, L, R, OUT: 'min> Sub<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Sub<R, Output = OUT> + Sub<&'min R, Output = OUT>,
    &'min L: Sub<R, Output = OUT> + Sub<&'min R, Output = OUT>, 
[src]

type Output = OUT

The resulting type after applying the - operator.

impl<'min, L, R> SubAssign<MaybeOwned<'min, R>> for MaybeOwned<'min, L> where
    L: Clone + SubAssign<R> + SubAssign<&'min R>, 
[src]

Auto Trait Implementations

impl<'a, T> RefUnwindSafe for MaybeOwned<'a, T> where
    T: RefUnwindSafe

impl<'a, T> Send for MaybeOwned<'a, T> where
    T: Send + Sync

impl<'a, T> Sync for MaybeOwned<'a, T> where
    T: Sync

impl<'a, T> Unpin for MaybeOwned<'a, T> where
    T: Unpin

impl<'a, T> UnwindSafe for MaybeOwned<'a, T> where
    T: RefUnwindSafe + UnwindSafe

Blanket Implementations

impl<T> Any for T where
    T: 'static + ?Sized
[src]

impl<T> Borrow<T> for T where
    T: ?Sized
[src]

impl<T> BorrowMut<T> for T where
    T: ?Sized
[src]

impl<T> From<!> for T[src]

impl<T> From<T> for T[src]

impl<T, U> Into<U> for T where
    U: From<T>, 
[src]

impl<T> ToOwned for T where
    T: Clone
[src]

type Owned = T

The resulting type after obtaining ownership.

impl<T> ToString for T where
    T: Display + ?Sized
[src]

impl<T, U> TryFrom<U> for T where
    U: Into<T>, 
[src]

type Error = Infallible

The type returned in the event of a conversion error.

impl<T, U> TryInto<U> for T where
    U: TryFrom<T>, 
[src]

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.