rustmax::std::borrow

Trait Borrow

1.0.0 · Source
pub trait Borrow<Borrowed>
where Borrowed: ?Sized,
{ // Required method fn borrow(&self) -> &Borrowed; }
Expand description

A trait for borrowing data.

In Rust, it is common to provide different representations of a type for different use cases. For instance, storage location and management for a value can be specifically chosen as appropriate for a particular use via pointer types such as Box<T> or Rc<T>. Beyond these generic wrappers that can be used with any type, some types provide optional facets providing potentially costly functionality. An example for such a type is String which adds the ability to extend a string to the basic str. This requires keeping additional information unnecessary for a simple, immutable string.

These types provide access to the underlying data through references to the type of that data. They are said to be ‘borrowed as’ that type. For instance, a Box<T> can be borrowed as T while a String can be borrowed as str.

Types express that they can be borrowed as some type T by implementing Borrow<T>, providing a reference to a T in the trait’s borrow method. A type is free to borrow as several different types. If it wishes to mutably borrow as the type, allowing the underlying data to be modified, it can additionally implement BorrowMut<T>.

Further, when providing implementations for additional traits, it needs to be considered whether they should behave identically to those of the underlying type as a consequence of acting as a representation of that underlying type. Generic code typically uses Borrow<T> when it relies on the identical behavior of these additional trait implementations. These traits will likely appear as additional trait bounds.

In particular Eq, Ord and Hash must be equivalent for borrowed and owned values: x.borrow() == y.borrow() should give the same result as x == y.

If generic code merely needs to work for all types that can provide a reference to related type T, it is often better to use AsRef<T> as more types can safely implement it.

§Examples

As a data collection, HashMap<K, V> owns both keys and values. If the key’s actual data is wrapped in a managing type of some kind, it should, however, still be possible to search for a value using a reference to the key’s data. For instance, if the key is a string, then it is likely stored with the hash map as a String, while it should be possible to search using a &str. Thus, insert needs to operate on a String while get needs to be able to use a &str.

Slightly simplified, the relevant parts of HashMap<K, V> look like this:

use std::borrow::Borrow;
use std::hash::Hash;

pub struct HashMap<K, V> {
    // fields omitted
}

impl<K, V> HashMap<K, V> {
    pub fn insert(&self, key: K, value: V) -> Option<V>
    where K: Hash + Eq
    {
        // ...
    }

    pub fn get<Q>(&self, k: &Q) -> Option<&V>
    where
        K: Borrow<Q>,
        Q: Hash + Eq + ?Sized
    {
        // ...
    }
}

The entire hash map is generic over a key type K. Because these keys are stored with the hash map, this type has to own the key’s data. When inserting a key-value pair, the map is given such a K and needs to find the correct hash bucket and check if the key is already present based on that K. It therefore requires K: Hash + Eq.

When searching for a value in the map, however, having to provide a reference to a K as the key to search for would require to always create such an owned value. For string keys, this would mean a String value needs to be created just for the search for cases where only a str is available.

Instead, the get method is generic over the type of the underlying key data, called Q in the method signature above. It states that K borrows as a Q by requiring that K: Borrow<Q>. By additionally requiring Q: Hash + Eq, it signals the requirement that K and Q have implementations of the Hash and Eq traits that produce identical results.

The implementation of get relies in particular on identical implementations of Hash by determining the key’s hash bucket by calling Hash::hash on the Q value even though it inserted the key based on the hash value calculated from the K value.

As a consequence, the hash map breaks if a K wrapping a Q value produces a different hash than Q. For instance, imagine you have a type that wraps a string but compares ASCII letters ignoring their case:

pub struct CaseInsensitiveString(String);

impl PartialEq for CaseInsensitiveString {
    fn eq(&self, other: &Self) -> bool {
        self.0.eq_ignore_ascii_case(&other.0)
    }
}

impl Eq for CaseInsensitiveString { }

Because two equal values need to produce the same hash value, the implementation of Hash needs to ignore ASCII case, too:

impl Hash for CaseInsensitiveString {
    fn hash<H: Hasher>(&self, state: &mut H) {
        for c in self.0.as_bytes() {
            c.to_ascii_lowercase().hash(state)
        }
    }
}

Can CaseInsensitiveString implement Borrow<str>? It certainly can provide a reference to a string slice via its contained owned string. But because its Hash implementation differs, it behaves differently from str and therefore must not, in fact, implement Borrow<str>. If it wants to allow others access to the underlying str, it can do that via AsRef<str> which doesn’t carry any extra requirements.

Required Methods§

1.0.0 · Source

fn borrow(&self) -> &Borrowed

Immutably borrows from an owned value.

§Examples
use std::borrow::Borrow;

fn check<T: Borrow<str>>(s: T) {
    assert_eq!("Hello", s.borrow());
}

let s = "Hello".to_string();

check(s);

let s = "Hello";

check(s);

Implementors§

Source§

impl Borrow<str> for InternalString

Source§

impl Borrow<str> for Key

Source§

impl Borrow<str> for Str

Source§

impl Borrow<str> for Id

Source§

impl Borrow<str> for HeaderName

Source§

impl Borrow<str> for Spanned<String>

1.0.0 · Source§

impl Borrow<str> for String

Source§

impl Borrow<BStr> for str

Source§

impl Borrow<BStr> for BString

Source§

impl Borrow<BStr> for String

Source§

impl Borrow<BStr> for Vec<u8>

Source§

impl Borrow<BStr> for [u8]

Source§

impl Borrow<Asn1BitStringRef> for Asn1BitString

Source§

impl Borrow<Asn1EnumeratedRef> for Asn1Enumerated

Source§

impl Borrow<Asn1GeneralizedTimeRef> for Asn1GeneralizedTime

Source§

impl Borrow<Asn1IntegerRef> for Asn1Integer

Source§

impl Borrow<Asn1ObjectRef> for Asn1Object

Source§

impl Borrow<Asn1OctetStringRef> for Asn1OctetString

Source§

impl Borrow<Asn1StringRef> for Asn1String

Source§

impl Borrow<Asn1TimeRef> for Asn1Time

Source§

impl Borrow<BigNumContextRef> for BigNumContext

Source§

impl Borrow<BigNumRef> for BigNum

Source§

impl Borrow<CipherCtxRef> for CipherCtx

Source§

impl Borrow<CmsContentInfoRef> for CmsContentInfo

Source§

impl Borrow<ConfRef> for Conf

Source§

impl Borrow<DsaSigRef> for DsaSig

Source§

impl Borrow<EcGroupRef> for EcGroup

Source§

impl Borrow<EcPointRef> for EcPoint

Source§

impl Borrow<EcdsaSigRef> for EcdsaSig

Source§

impl Borrow<LibCtxRef> for LibCtx

Source§

impl Borrow<MdCtxRef> for MdCtx

Source§

impl Borrow<OcspBasicResponseRef> for OcspBasicResponse

Source§

impl Borrow<OcspCertIdRef> for OcspCertId

Source§

impl Borrow<OcspOneReqRef> for OcspOneReq

Source§

impl Borrow<OcspRequestRef> for OcspRequest

Source§

impl Borrow<OcspResponseRef> for OcspResponse

Source§

impl Borrow<Pkcs7Ref> for Pkcs7

Source§

impl Borrow<Pkcs7SignedRef> for Pkcs7Signed

Source§

impl Borrow<Pkcs7SignerInfoRef> for Pkcs7SignerInfo

Source§

impl Borrow<Pkcs12Ref> for Pkcs12

Source§

impl Borrow<ProviderRef> for Provider

Source§

impl Borrow<SrtpProtectionProfileRef> for SrtpProtectionProfile

Source§

impl Borrow<SslContextRef> for SslContext

Source§

impl Borrow<SslRef> for Ssl

Source§

impl Borrow<SslSessionRef> for SslSession

Source§

impl Borrow<OpensslStringRef> for OpensslString

Source§

impl Borrow<X509StoreBuilderRef> for X509StoreBuilder

Source§

impl Borrow<X509StoreRef> for X509Store

Source§

impl Borrow<AccessDescriptionRef> for AccessDescription

Source§

impl Borrow<DistPointNameRef> for DistPointName

Source§

impl Borrow<DistPointRef> for DistPoint

Source§

impl Borrow<GeneralNameRef> for GeneralName

Source§

impl Borrow<X509AlgorithmRef> for X509Algorithm

Source§

impl Borrow<X509CrlRef> for X509Crl

Source§

impl Borrow<X509ExtensionRef> for X509Extension

Source§

impl Borrow<X509NameEntryRef> for X509NameEntry

Source§

impl Borrow<X509NameRef> for X509Name

Source§

impl Borrow<X509ObjectRef> for X509Object

Source§

impl Borrow<X509Ref> for X509

Source§

impl Borrow<X509ReqRef> for X509Req

Source§

impl Borrow<X509RevokedRef> for X509Revoked

Source§

impl Borrow<X509StoreContextRef> for X509StoreContext

Source§

impl Borrow<X509VerifyParamRef> for X509VerifyParam

Source§

impl Borrow<BStr> for Vec<u8>

Source§

impl Borrow<Bytes> for Vec<u8>

1.3.0 · Source§

impl Borrow<CStr> for CString

Source§

impl Borrow<OsStr> for OsStr

1.0.0 · Source§

impl Borrow<OsStr> for OsString

1.0.0 · Source§

impl Borrow<Path> for PathBuf

Source§

impl Borrow<[u8]> for BStr

Source§

impl Borrow<[u8]> for BString

Source§

impl Borrow<[u8]> for BytesMut

Source§

impl Borrow<[u8]> for Bytes

1.0.0 · Source§

impl<'a, B> Borrow<B> for Cow<'a, B>
where B: ToOwned + ?Sized,

Source§

impl<A> Borrow<[<A as Array>::Item]> for SmallVec<A>
where A: Array,

Source§

impl<T> Borrow<DhRef<T>> for Dh<T>

Source§

impl<T> Borrow<DsaRef<T>> for Dsa<T>

Source§

impl<T> Borrow<EcKeyRef<T>> for EcKey<T>

Source§

impl<T> Borrow<PKeyRef<T>> for PKey<T>

Source§

impl<T> Borrow<PkeyCtxRef<T>> for PkeyCtx<T>

Source§

impl<T> Borrow<RsaRef<T>> for Rsa<T>

Source§

impl<T> Borrow<StackRef<T>> for Stack<T>
where T: Stackable,

Source§

impl<T> Borrow<X509LookupMethodRef<T>> for X509LookupMethod<T>

Source§

impl<T> Borrow<X509LookupRef<T>> for X509Lookup<T>

1.0.0 · Source§

impl<T> Borrow<T> for &T
where T: ?Sized,

1.0.0 · Source§

impl<T> Borrow<T> for &mut T
where T: ?Sized,

Source§

impl<T> Borrow<T> for Owned<T>
where T: Pointable + ?Sized,

1.0.0 · Source§

impl<T> Borrow<T> for T
where T: ?Sized,

1.0.0 · Source§

impl<T, A> Borrow<[T]> for Vec<T, A>
where A: Allocator,

1.1.0 · Source§

impl<T, A> Borrow<T> for Box<T, A>
where A: Allocator, T: ?Sized,

1.0.0 · Source§

impl<T, A> Borrow<T> for Rc<T, A>
where A: Allocator, T: ?Sized,

1.0.0 · Source§

impl<T, A> Borrow<T> for Arc<T, A>
where A: Allocator, T: ?Sized,

Source§

impl<T, N> Borrow<[T]> for GenericArray<T, N>
where N: ArrayLength<T>,

Source§

impl<T, const CAP: usize> Borrow<[T]> for ArrayVec<T, CAP>

1.4.0 · Source§

impl<T, const N: usize> Borrow<[T]> for [T; N]

Source§

impl<const CAP: usize> Borrow<str> for ArrayString<CAP>