-
Notifications
You must be signed in to change notification settings - Fork 13.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Improve documentation for Borrow #46518
Merged
Merged
Changes from 6 commits
Commits
Show all changes
9 commits
Select commit
Hold shift + click to select a range
cba5f6b
Rewrite Borrow's trait documentation.
partim c4ea700
Remove trailing white space.
partim 85e8a9b
Include feedback and try to make examples build on all channels.
partim fc6c638
Fix documentation links.
partim 7ae7e53
New introduction and revised hash map explanation.
partim 44be054
Further refinement of Borrow documentation.
partim 5bef034
Bring back the phrase 'borrowing as' for what Borrow does.
partim d664b89
Rewrite the documentation for BorrowMut.
partim 13d94d6
Fix formatting.
partim File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -12,26 +12,161 @@ | |
|
||
#![stable(feature = "rust1", since = "1.0.0")] | ||
|
||
/// A trait for borrowing data. | ||
/// A trait identifying how borrowed data behaves. | ||
/// | ||
/// In general, there may be several ways to "borrow" a piece of data. The | ||
/// typical ways of borrowing a type `T` are `&T` (a shared borrow) and `&mut T` | ||
/// (a mutable borrow). But types like `Vec<T>` provide additional kinds of | ||
/// borrows: the borrowed slices `&[T]` and `&mut [T]`. | ||
/// 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. | ||
/// | ||
/// When writing generic code, it is often desirable to abstract over all ways | ||
/// of borrowing data from a given type. That is the role of the `Borrow` | ||
/// trait: if `T: Borrow<U>`, then `&U` can be borrowed from `&T`. A given | ||
/// type can be borrowed as multiple different types. In particular, `Vec<T>: | ||
/// Borrow<Vec<T>>` and `Vec<T>: Borrow<[T]>`. | ||
/// These types signal that they are a specialized representation of a basic | ||
/// type `T` by implementing `Borrow<T>`. The method `borrow` provides a way | ||
/// to convert a reference to the type into a reference to this basic type | ||
/// `T`. | ||
/// | ||
/// If you are implementing `Borrow` and both `Self` and `Borrowed` implement | ||
/// `Hash`, `Eq`, and/or `Ord`, they must produce the same result. | ||
/// Further, when providing implementations for additional traits, it needs | ||
/// to be considered whether they should behave identical to those of the | ||
/// underlying type as a consequence of acting as a representation of that | ||
/// underlying type. | ||
/// | ||
/// `Borrow` is very similar to, but different than, `AsRef`. See | ||
/// [the book][book] for more. | ||
/// Generic code typically uses `Borrow<T>` when it not only needs access | ||
/// to a reference of the underlying type but relies on the identical | ||
/// behavior of these additional trait implementations. These traits are | ||
/// likely to appear as additional trait bounds. | ||
/// | ||
/// 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. | ||
/// | ||
/// If a type implementing `Borrow<T>` also wishes to allow mutable access | ||
/// to the underlying type `T`, it can do so by implementing the companion | ||
/// trait [`BorrowMut`]. | ||
/// | ||
/// Note also that it is perfectly fine for a single type to have multiple | ||
/// implementations of `Borrow<T>` for different `T`s. In fact, a blanket | ||
/// implementation lets every type be at least a borrow of itself. | ||
/// | ||
/// [`AsRef<T>`]: ../../std/convert/trait.AsRef.html | ||
/// [`BorrowMut`]: trait.BorrowMut.html | ||
/// [`Box<T>`]: ../../std/boxed/struct.Box.html | ||
/// [`Mutex<T>`]: ../../std/sync/struct.Mutex.html | ||
/// [`Rc<T>`]: ../../std/rc/struct.Rc.html | ||
/// [`str`]: ../../std/primitive.str.html | ||
/// [`String`]: ../../std/string/struct.String.html | ||
/// | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. can you remove one of these blank lines please? |
||
/// # 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`][`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> { | ||
/// # marker: ::std::marker::PhantomData<(K, V)>, | ||
/// // fields omitted | ||
/// } | ||
/// | ||
/// impl<K, V> HashMap<K, V> { | ||
/// pub fn insert(&self, key: K, value: V) -> Option<V> | ||
/// where K: Hash + Eq | ||
/// { | ||
/// # unimplemented!() | ||
/// // ... | ||
/// } | ||
/// | ||
/// pub fn get<Q>(&self, k: &Q) -> Option<&V> | ||
/// where | ||
/// K: Borrow<Q>, | ||
/// Q: Hash + Eq + ?Sized | ||
/// { | ||
/// # unimplemented!() | ||
/// // ... | ||
/// } | ||
/// } | ||
/// ``` | ||
/// | ||
/// 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` is a | ||
/// representation of `Q` by requiring that `K: Borrow<Q>`. By additionally | ||
/// requiring `Q: Hash + Eq`, it demands 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 reflect that, too: | ||
/// | ||
/// ``` | ||
/// # use std::hash::{Hash, Hasher}; | ||
/// # pub struct CaseInsensitiveString(String); | ||
/// 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. | ||
/// | ||
/// [`Hash`]: ../../std/hash/trait.Hash.html | ||
/// [`HashMap<K, V>`]: ../../std/collections/struct.HashMap.html | ||
/// [`String`]: ../../std/string/struct.String.html | ||
/// [`str`]: ../../std/primitive.str.html | ||
/// | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. can you remove this line please? |
||
/// [book]: ../../book/first-edition/borrow-and-asref.html | ||
#[stable(feature = "rust1", since = "1.0.0")] | ||
pub trait Borrow<Borrowed: ?Sized> { | ||
/// Immutably borrows from an owned value. | ||
|
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I find this sentence confusing. How about:
"Implementors of
Borrow<Borrowed>
can be borrowed asBorrowed
."Now, that is a lot of borrowing, so perhaps the following reads better?
"Implementors of
Borrow<T>
can be borrowed asT
."Just an idea. :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That is true for
AsRef<T>
, too, though. I don’t like the sentence either, but if at all possible in one sentence, the summary should express those strict guarantees implied byBorrow<T>
. Or at least be enough of a tease that people will read on.