Skip to content

Commit

Permalink
Document the where keyword
Browse files Browse the repository at this point in the history
  • Loading branch information
poliorcetics committed Jul 3, 2020
1 parent f844ea1 commit 837a761
Showing 1 changed file with 135 additions and 2 deletions.
137 changes: 135 additions & 2 deletions src/libstd/keyword_docs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1651,9 +1651,142 @@ mod use_keyword {}
//
/// Add constraints that must be upheld to use an item.
///
/// The documentation for this keyword is [not yet complete]. Pull requests welcome!
/// `where` allows specifying constraints on lifetime and generic parameters.
/// The [RFC] introducing `where` contains detailed informations about the
/// keyword.
///
/// [not yet complete]: https://github.com/rust-lang/rust/issues/34601
/// # Examples
///
/// `where` can be used for constraints with traits:
///
/// ```rust
/// fn new<T: Default>() -> T {
/// T::default()
/// }
///
/// fn new_where<T>() -> T
/// where
/// T: Default,
/// {
/// T::default()
/// }
///
/// assert_eq!(0.0, new());
/// assert_eq!(0.0, new_where());
///
/// assert_eq!(0, new());
/// assert_eq!(0, new_where());
/// ```
///
/// `where` can also be used for lifetimes.
///
/// This compiles because the lifetime of `longer` is superior to the lifetime
/// of `shorter`, thus the constraint is respected:
///
/// ```rust
/// fn select<'a, 'b: 'a>(s1: &'a str, s2: &'b str, second: bool) -> &'a str {
/// if second {
/// s2
/// } else {
/// s1
/// }
/// }
///
/// fn select_where<'a, 'b>(s1: &'a str, s2: &'b str, second: bool) -> &'a str
/// where
/// 'b: 'a,
/// {
/// if second {
/// s2
/// } else {
/// s1
/// }
/// }
///
/// let outer = String::from("Long living ref");
/// let longer = &outer;
/// {
/// let inner = String::from("Long living ref");
/// let shorter = &inner;
///
/// assert_eq!(select(shorter, longer, false), shorter);
/// assert_eq!(select(shorter, longer, true), longer);
///
/// assert_eq!(select_where(shorter, longer, false), shorter);
/// assert_eq!(select_where(shorter, longer, true), longer);
/// }
/// ```
///
/// On the other hand, this will not compile: `shorter` does not have a lifetime
/// that respects the constraint imposed by the `select` and `select_where`
/// functions.
///
/// ```rust,compile_fail,E0597
/// # fn select<'a, 'b: 'a>(s1: &'a str, s2: &'b str, second: bool) -> &'a str {
/// # if second {
/// # s2
/// # } else {
/// # s1
/// # }
/// # }
/// #
/// # fn select_where<'a, 'b>(s1: &'a str, s2: &'b str, second: bool) -> &'a str
/// # where
/// # 'b: 'a,
/// # {
/// # if second {
/// # s2
/// # } else {
/// # s1
/// # }
/// # }
/// let outer = String::from("Long living ref");
/// let longer = &outer;
/// let res1;
/// let res2;
/// {
/// let inner = String::from("Long living ref");
/// let shorter = &inner;
///
/// res1 = select(longer, shorter, false);
/// res2 = select_where(longer, shorter, false);
/// }
/// assert_eq!(res1, &outer);
/// assert_eq!(res2, &outer);
/// ```
///
/// `where` can also be used to express more complicated constraints that cannot
/// be written with the `<T: Trait>` syntax:
///
/// ```rust
/// fn first_or_default<I>(mut i: I) -> I::Item
/// where
/// I: Iterator,
/// I::Item: Default,
/// {
/// i.next().unwrap_or_else(I::Item::default)
/// }
///
/// assert_eq!(first_or_default(vec![1, 2, 3].into_iter()), 1);
/// assert_eq!(first_or_default(Vec::<i32>::new().into_iter()), 0);
/// ```
///
/// `where` is available anywhere generic and lifetime parameters are available:
///
/// ```rust
/// # #![allow(dead_code)]
/// // The Cow type from the standard library uses where to impose constraints
/// // on its parameters.
/// pub enum Cow<'a, B>
/// where
/// B: 'a + ToOwned + ?Sized,
/// {
/// Borrowed(&'a B),
/// Owned(<B as ToOwned>::Owned),
/// }
/// ```
///
/// [RFC]: https://github.com/rust-lang/rfcs/blob/master/text/0135-where.md
mod where_keyword {}

// 2018 Edition keywords
Expand Down

0 comments on commit 837a761

Please sign in to comment.