core/
borrow.rs

1//! Utilities for working with borrowed data.
2
3#![stable(feature = "rust1", since = "1.0.0")]
4
5/// A trait for borrowing data.
6///
7/// In Rust, it is common to provide different representations of a type for
8/// different use cases. For instance, storage location and management for a
9/// value can be specifically chosen as appropriate for a particular use via
10/// pointer types such as [`Box<T>`] or [`Rc<T>`]. Beyond these generic
11/// wrappers that can be used with any type, some types provide optional
12/// facets providing potentially costly functionality. An example for such a
13/// type is [`String`] which adds the ability to extend a string to the basic
14/// [`str`]. This requires keeping additional information unnecessary for a
15/// simple, immutable string.
16///
17/// These types provide access to the underlying data through references
18/// to the type of that data. They are said to be ‘borrowed as’ that type.
19/// For instance, a [`Box<T>`] can be borrowed as `T` while a [`String`]
20/// can be borrowed as `str`.
21///
22/// Types express that they can be borrowed as some type `T` by implementing
23/// `Borrow<T>`, providing a reference to a `T` in the trait’s
24/// [`borrow`] method. A type is free to borrow as several different types.
25/// If it wishes to mutably borrow as the type, allowing the underlying data
26/// to be modified, it can additionally implement [`BorrowMut<T>`].
27///
28/// Further, when providing implementations for additional traits, it needs
29/// to be considered whether they should behave identically to those of the
30/// underlying type as a consequence of acting as a representation of that
31/// underlying type. Generic code typically uses `Borrow<T>` when it relies
32/// on the identical behavior of these additional trait implementations.
33/// These traits will likely appear as additional trait bounds.
34///
35/// In particular `Eq`, `Ord` and `Hash` must be equivalent for
36/// borrowed and owned values: `x.borrow() == y.borrow()` should give the
37/// same result as `x == y`.
38///
39/// If generic code merely needs to work for all types that can
40/// provide a reference to related type `T`, it is often better to use
41/// [`AsRef<T>`] as more types can safely implement it.
42///
43/// [`Box<T>`]: ../../std/boxed/struct.Box.html
44/// [`Mutex<T>`]: ../../std/sync/struct.Mutex.html
45/// [`Rc<T>`]: ../../std/rc/struct.Rc.html
46/// [`String`]: ../../std/string/struct.String.html
47/// [`borrow`]: Borrow::borrow
48///
49/// # Examples
50///
51/// As a data collection, [`HashMap<K, V>`] owns both keys and values. If
52/// the key’s actual data is wrapped in a managing type of some kind, it
53/// should, however, still be possible to search for a value using a
54/// reference to the key’s data. For instance, if the key is a string, then
55/// it is likely stored with the hash map as a [`String`], while it should
56/// be possible to search using a [`&str`][`str`]. Thus, `insert` needs to
57/// operate on a `String` while `get` needs to be able to use a `&str`.
58///
59/// Slightly simplified, the relevant parts of `HashMap<K, V>` look like
60/// this:
61///
62/// ```
63/// use std::borrow::Borrow;
64/// use std::hash::Hash;
65///
66/// pub struct HashMap<K, V> {
67///     # marker: ::std::marker::PhantomData<(K, V)>,
68///     // fields omitted
69/// }
70///
71/// impl<K, V> HashMap<K, V> {
72///     pub fn insert(&self, key: K, value: V) -> Option<V>
73///     where K: Hash + Eq
74///     {
75///         # unimplemented!()
76///         // ...
77///     }
78///
79///     pub fn get<Q>(&self, k: &Q) -> Option<&V>
80///     where
81///         K: Borrow<Q>,
82///         Q: Hash + Eq + ?Sized
83///     {
84///         # unimplemented!()
85///         // ...
86///     }
87/// }
88/// ```
89///
90/// The entire hash map is generic over a key type `K`. Because these keys
91/// are stored with the hash map, this type has to own the key’s data.
92/// When inserting a key-value pair, the map is given such a `K` and needs
93/// to find the correct hash bucket and check if the key is already present
94/// based on that `K`. It therefore requires `K: Hash + Eq`.
95///
96/// When searching for a value in the map, however, having to provide a
97/// reference to a `K` as the key to search for would require to always
98/// create such an owned value. For string keys, this would mean a `String`
99/// value needs to be created just for the search for cases where only a
100/// `str` is available.
101///
102/// Instead, the `get` method is generic over the type of the underlying key
103/// data, called `Q` in the method signature above. It states that `K`
104/// borrows as a `Q` by requiring that `K: Borrow<Q>`. By additionally
105/// requiring `Q: Hash + Eq`, it signals the requirement that `K` and `Q`
106/// have implementations of the `Hash` and `Eq` traits that produce identical
107/// results.
108///
109/// The implementation of `get` relies in particular on identical
110/// implementations of `Hash` by determining the key’s hash bucket by calling
111/// `Hash::hash` on the `Q` value even though it inserted the key based on
112/// the hash value calculated from the `K` value.
113///
114/// As a consequence, the hash map breaks if a `K` wrapping a `Q` value
115/// produces a different hash than `Q`. For instance, imagine you have a
116/// type that wraps a string but compares ASCII letters ignoring their case:
117///
118/// ```
119/// pub struct CaseInsensitiveString(String);
120///
121/// impl PartialEq for CaseInsensitiveString {
122///     fn eq(&self, other: &Self) -> bool {
123///         self.0.eq_ignore_ascii_case(&other.0)
124///     }
125/// }
126///
127/// impl Eq for CaseInsensitiveString { }
128/// ```
129///
130/// Because two equal values need to produce the same hash value, the
131/// implementation of `Hash` needs to ignore ASCII case, too:
132///
133/// ```
134/// # use std::hash::{Hash, Hasher};
135/// # pub struct CaseInsensitiveString(String);
136/// impl Hash for CaseInsensitiveString {
137///     fn hash<H: Hasher>(&self, state: &mut H) {
138///         for c in self.0.as_bytes() {
139///             c.to_ascii_lowercase().hash(state)
140///         }
141///     }
142/// }
143/// ```
144///
145/// Can `CaseInsensitiveString` implement `Borrow<str>`? It certainly can
146/// provide a reference to a string slice via its contained owned string.
147/// But because its `Hash` implementation differs, it behaves differently
148/// from `str` and therefore must not, in fact, implement `Borrow<str>`.
149/// If it wants to allow others access to the underlying `str`, it can do
150/// that via `AsRef<str>` which doesn’t carry any extra requirements.
151///
152/// [`Hash`]: crate::hash::Hash
153/// [`HashMap<K, V>`]: ../../std/collections/struct.HashMap.html
154/// [`String`]: ../../std/string/struct.String.html
155#[stable(feature = "rust1", since = "1.0.0")]
156#[rustc_diagnostic_item = "Borrow"]
157pub trait Borrow<Borrowed: ?Sized> {
158    /// Immutably borrows from an owned value.
159    ///
160    /// # Examples
161    ///
162    /// ```
163    /// use std::borrow::Borrow;
164    ///
165    /// fn check<T: Borrow<str>>(s: T) {
166    ///     assert_eq!("Hello", s.borrow());
167    /// }
168    ///
169    /// let s = "Hello".to_string();
170    ///
171    /// check(s);
172    ///
173    /// let s = "Hello";
174    ///
175    /// check(s);
176    /// ```
177    #[stable(feature = "rust1", since = "1.0.0")]
178    fn borrow(&self) -> &Borrowed;
179}
180
181/// A trait for mutably borrowing data.
182///
183/// As a companion to [`Borrow<T>`] this trait allows a type to borrow as
184/// an underlying type by providing a mutable reference. See [`Borrow<T>`]
185/// for more information on borrowing as another type.
186#[stable(feature = "rust1", since = "1.0.0")]
187#[rustc_diagnostic_item = "BorrowMut"]
188pub trait BorrowMut<Borrowed: ?Sized>: Borrow<Borrowed> {
189    /// Mutably borrows from an owned value.
190    ///
191    /// # Examples
192    ///
193    /// ```
194    /// use std::borrow::BorrowMut;
195    ///
196    /// fn check<T: BorrowMut<[i32]>>(mut v: T) {
197    ///     assert_eq!(&mut [1, 2, 3], v.borrow_mut());
198    /// }
199    ///
200    /// let v = vec![1, 2, 3];
201    ///
202    /// check(v);
203    /// ```
204    #[stable(feature = "rust1", since = "1.0.0")]
205    fn borrow_mut(&mut self) -> &mut Borrowed;
206}
207
208#[stable(feature = "rust1", since = "1.0.0")]
209impl<T: ?Sized> Borrow<T> for T {
210    #[rustc_diagnostic_item = "noop_method_borrow"]
211    fn borrow(&self) -> &T {
212        self
213    }
214}
215
216#[stable(feature = "rust1", since = "1.0.0")]
217impl<T: ?Sized> BorrowMut<T> for T {
218    fn borrow_mut(&mut self) -> &mut T {
219        self
220    }
221}
222
223#[stable(feature = "rust1", since = "1.0.0")]
224impl<T: ?Sized> Borrow<T> for &T {
225    fn borrow(&self) -> &T {
226        &**self
227    }
228}
229
230#[stable(feature = "rust1", since = "1.0.0")]
231impl<T: ?Sized> Borrow<T> for &mut T {
232    fn borrow(&self) -> &T {
233        &**self
234    }
235}
236
237#[stable(feature = "rust1", since = "1.0.0")]
238impl<T: ?Sized> BorrowMut<T> for &mut T {
239    fn borrow_mut(&mut self) -> &mut T {
240        &mut **self
241    }
242}
pFad - Phonifier reborn

Pfad - The Proxy pFad of © 2024 Garber Painting. All rights reserved.

Note: This service is not intended for secure transactions such as banking, social media, email, or purchasing. Use at your own risk. We assume no liability whatsoever for broken pages.


Alternative Proxies:

Alternative Proxy

pFad Proxy

pFad v3 Proxy

pFad v4 Proxy