From bd417a4cab6f33b4a7e4aca6b1eda7fbde9aa713 Mon Sep 17 00:00:00 2001 From: Taylor Cramer Date: Mon, 21 Sep 2020 23:06:14 +0200 Subject: [PATCH 1/6] Add `take_...` functions to slices --- library/core/src/ops/mod.rs | 3 + library/core/src/ops/range.rs | 18 +++ library/core/src/slice/mod.rs | 241 +++++++++++++++++++++++++++++++++- library/core/tests/lib.rs | 1 + library/core/tests/slice.rs | 104 +++++++++++++++ 5 files changed, 366 insertions(+), 1 deletion(-) diff --git a/library/core/src/ops/mod.rs b/library/core/src/ops/mod.rs index 354ad6b7b7333..77da3e6318a8f 100644 --- a/library/core/src/ops/mod.rs +++ b/library/core/src/ops/mod.rs @@ -181,6 +181,9 @@ pub use self::range::{Range, RangeFrom, RangeFull, RangeTo}; #[stable(feature = "inclusive_range", since = "1.26.0")] pub use self::range::{Bound, RangeBounds, RangeInclusive, RangeToInclusive}; +#[unstable(feature = "one_sided_range", issue = "69780")] +pub use self::range::OneSidedRange; + #[unstable(feature = "try_trait", issue = "42327")] pub use self::r#try::Try; diff --git a/library/core/src/ops/range.rs b/library/core/src/ops/range.rs index 1d67e65e51f5f..3a057d4146723 100644 --- a/library/core/src/ops/range.rs +++ b/library/core/src/ops/range.rs @@ -1004,3 +1004,21 @@ impl RangeBounds for RangeToInclusive<&T> { Included(self.end) } } + +/// `OneSidedRange` is implemented by Rust's built-in range types which +/// are unbounded on one side. For example, `a..`, `..b` and `..=c` implement +/// `OneSidedRange`, but `..`, `d..e`, and `f..=g` do not. +/// +/// Types which implement `OneSidedRange` must return `Bound::Unbounded` +/// from exactly one of `RangeBounds::start_bound` and `RangeBounds::end_bound`. +#[unstable(feature = "one_sided_range", issue = "69780")] +pub trait OneSidedRange: RangeBounds {} + +#[unstable(feature = "one_sided_range", issue = "69780")] +impl OneSidedRange for RangeTo where Self: RangeBounds {} + +#[unstable(feature = "one_sided_range", issue = "69780")] +impl OneSidedRange for RangeFrom where Self: RangeBounds {} + +#[unstable(feature = "one_sided_range", issue = "69780")] +impl OneSidedRange for RangeToInclusive where Self: RangeBounds {} diff --git a/library/core/src/slice/mod.rs b/library/core/src/slice/mod.rs index 79ae1d5829a24..978c351d34276 100644 --- a/library/core/src/slice/mod.rs +++ b/library/core/src/slice/mod.rs @@ -12,7 +12,7 @@ use crate::cmp::Ordering::{self, Equal, Greater, Less}; use crate::marker::Copy; use crate::mem; use crate::num::NonZeroUsize; -use crate::ops::{FnMut, Range, RangeBounds}; +use crate::ops::{Bound, FnMut, OneSidedRange, Range, RangeBounds}; use crate::option::Option; use crate::option::Option::{None, Some}; use crate::ptr; @@ -73,6 +73,24 @@ pub use sort::heapsort; #[stable(feature = "slice_get_slice", since = "1.28.0")] pub use index::SliceIndex; +/// Calculates the direction and split point of a one-sided range. +/// +/// Helper for `take` and `take_mut` which returns a boolean +/// indicating whether the front of the split is being taken +/// (as opposed to the back), as well as a number indicating the +/// index at which to split. Returns `None` if the split index would +/// overflow `usize`. +#[inline] +fn take_split_point(range: impl OneSidedRange) -> Option<(bool, usize)> { + Some(match (range.start_bound(), range.end_bound()) { + (Bound::Unbounded, Bound::Excluded(i)) => (true, *i), + (Bound::Unbounded, Bound::Included(i)) => (true, i.checked_add(1)?), + (Bound::Excluded(i), Bound::Unbounded) => (false, i.checked_add(1)?), + (Bound::Included(i), Bound::Unbounded) => (false, *i), + _ => unreachable!(), + }) +} + #[lang = "slice"] #[cfg(not(test))] impl [T] { @@ -3169,6 +3187,227 @@ impl [T] { left } + + /// Removes and returns the portion of the slice specified by `range`. + /// + /// If the provided `range` starts or ends outside of the slice, + /// `None` is returned and the slice is not modified. + /// + /// # Examples + /// + /// Taking the first three items from a slice (via `..3`): + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &[_] = &['a', 'b', 'c', 'd']; + /// let mut first_three = slice.take(..3).unwrap(); + /// + /// assert_eq!(slice, &['d']); + /// assert_eq!(first_three, &['a', 'b', 'c']); + /// ``` + /// + /// Taking the tail of a slice starting at index two (via `2..`): + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &[_] = &['a', 'b', 'c', 'd']; + /// let mut tail = slice.take(2..).unwrap(); + /// + /// assert_eq!(slice, &['a', 'b']); + /// assert_eq!(tail, &['c', 'd']); + /// ``` + /// + /// Getting `None` when `range` starts or ends outside of the slice: + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &[_] = &['a', 'b', 'c', 'd']; + /// + /// assert_eq!(None, slice.take(5..)); + /// assert_eq!(None, slice.take(..5)); + /// assert_eq!(None, slice.take(..=4)); + /// let expected: &[char] = &['a', 'b', 'c', 'd']; + /// assert_eq!(Some(expected), slice.take(..4)); + /// ``` + #[inline] + #[unstable(feature = "slice_take", issue = "62280")] + pub fn take<'a, R: OneSidedRange>(self: &mut &'a Self, range: R) -> Option<&'a Self> { + let (taking_front, split_index) = take_split_point(range)?; + if split_index > self.len() { + return None; + } + let original = crate::mem::take(self); + let (front, back) = original.split_at(split_index); + if taking_front { + *self = back; + Some(front) + } else { + *self = front; + Some(back) + } + } + + /// Removes and returns the portion of the mutable slice specified by `range`. + /// + /// If the provided `range` starts or ends outside of the slice, + /// `None` is returned and the slice is not modified. + /// + /// # Examples + /// + /// Taking the first three items from a slice (via `..3`): + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &mut [_] = &mut ['a', 'b', 'c', 'd']; + /// let mut first_three = slice.take_mut(..3).unwrap(); + /// + /// assert_eq!(slice, &mut ['d']); + /// assert_eq!(first_three, &mut ['a', 'b', 'c']); + /// ``` + /// + /// Taking the tail of a slice starting at index two (via `2..`): + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &mut [_] = &mut ['a', 'b', 'c', 'd']; + /// let mut tail = slice.take_mut(2..).unwrap(); + /// + /// assert_eq!(slice, &mut ['a', 'b']); + /// assert_eq!(tail, &mut ['c', 'd']); + /// ``` + /// + /// Getting `None` when `range` starts or ends outside of the slice: + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &mut [_] = &mut ['a', 'b', 'c', 'd']; + /// + /// assert_eq!(None, slice.take_mut(5..)); + /// assert_eq!(None, slice.take_mut(..5)); + /// assert_eq!(None, slice.take_mut(..=4)); + /// let expected: &mut [_] = &mut ['a', 'b', 'c', 'd']; + /// assert_eq!(Some(expected), slice.take_mut(..4)); + /// ``` + #[inline] + #[unstable(feature = "slice_take", issue = "62280")] + pub fn take_mut<'a, R: OneSidedRange>( + self: &mut &'a mut Self, + range: R, + ) -> Option<&'a mut Self> { + let (taking_front, split_index) = take_split_point(range)?; + if split_index > self.len() { + return None; + } + let original = crate::mem::take(self); + let (front, back) = original.split_at_mut(split_index); + if taking_front { + *self = back; + Some(front) + } else { + *self = front; + Some(back) + } + } + + /// Takes the first element out of the slice. + /// + /// Returns a reference pointing to the first element of the old slice. + /// + /// Returns `None` if the slice is empty. + /// + /// # Examples + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &[_] = &['a', 'b', 'c']; + /// let first = slice.take_first().unwrap(); + /// + /// assert_eq!(slice, &['b', 'c']); + /// assert_eq!(first, &'a'); + /// ``` + #[inline] + #[unstable(feature = "slice_take", issue = "62280")] + pub fn take_first<'a>(self: &mut &'a Self) -> Option<&'a T> { + self.take(..=0).map(|res| &res[0]) + } + + /// Takes the first element out of the mutable slice. + /// + /// Returns a mutable reference pointing to the first element of the old slice. + /// + /// Returns `None` if the slice is empty. + /// + /// # Examples + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &mut [_] = &mut ['a', 'b', 'c']; + /// let first = slice.take_first_mut().unwrap(); + /// *first = 'd'; + /// + /// assert_eq!(slice, &['b', 'c']); + /// assert_eq!(first, &'d'); + /// ``` + #[inline] + #[unstable(feature = "slice_take", issue = "62280")] + pub fn take_first_mut<'a>(self: &mut &'a mut Self) -> Option<&'a mut T> { + self.take_mut(..=0).map(|res| &mut res[0]) + } + + /// Takes the last element out of the slice. + /// + /// Returns a reference pointing to the last element of the old slice. + /// + /// Returns `None` if the slice is empty. + /// + /// # Examples + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &[_] = &['a', 'b', 'c']; + /// let last = slice.take_last().unwrap(); + /// + /// assert_eq!(slice, &['a', 'b']); + /// assert_eq!(last, &'c'); + /// ``` + #[inline] + #[unstable(feature = "slice_take", issue = "62280")] + pub fn take_last<'a>(self: &mut &'a Self) -> Option<&'a T> { + self.take((self.len() - 1)..).map(|res| &res[0]) + } + + /// Takes the last element out of the mutable slice. + /// + /// Returns a mutable reference pointing to the last element of the old slice. + /// + /// Returns `None` if the slice is empty. + /// + /// # Examples + /// + /// ``` + /// #![feature(slice_take)] + /// + /// let mut slice: &mut [_] = &mut ['a', 'b', 'c']; + /// let last = slice.take_last_mut().unwrap(); + /// *last = 'd'; + /// + /// assert_eq!(slice, &['a', 'b']); + /// assert_eq!(last, &'d'); + /// ``` + #[inline] + #[unstable(feature = "slice_take", issue = "62280")] + pub fn take_last_mut<'a>(self: &mut &'a mut Self) -> Option<&'a mut T> { + self.take_mut((self.len() - 1)..).map(|res| &mut res[0]) + } } #[stable(feature = "rust1", since = "1.0.0")] diff --git a/library/core/tests/lib.rs b/library/core/tests/lib.rs index 0c4ce867f542d..51987de0398d3 100644 --- a/library/core/tests/lib.rs +++ b/library/core/tests/lib.rs @@ -39,6 +39,7 @@ #![feature(try_trait)] #![feature(slice_internals)] #![feature(slice_partition_dedup)] +#![feature(slice_take)] #![feature(int_error_matching)] #![feature(array_value_iter)] #![feature(iter_advance_by)] diff --git a/library/core/tests/slice.rs b/library/core/tests/slice.rs index 9ccc5a08dcbea..8f5386b2ba0e6 100644 --- a/library/core/tests/slice.rs +++ b/library/core/tests/slice.rs @@ -2038,3 +2038,107 @@ fn test_slice_run_destructors() { assert_eq!(x.get(), 1); } + +macro_rules! take_tests { + (slice: &$slice:expr, $($tts:tt)*) => { + take_tests!(ty: &[_], slice: &$slice, $($tts)*); + }; + (slice: &mut $slice:expr, $($tts:tt)*) => { + take_tests!(ty: &mut [_], slice: &mut $slice, $($tts)*); + }; + (ty: $ty:ty, slice: $slice:expr, method: $method:ident, $(($test_name:ident, ($($args:expr),*), $output:expr, $remaining:expr),)*) => { + $( + #[test] + fn $test_name() { + let mut slice: $ty = $slice; + assert_eq!($output, slice.$method($($args)*)); + let remaining: $ty = $remaining; + assert_eq!(remaining, slice); + } + )* + }; +} + +take_tests! { + slice: &[0, 1, 2, 3], method: take, + (take_in_bounds_range_to, (..1), Some(&[0] as _), &[1, 2, 3]), + (take_in_bounds_range_to_inclusive, (..=0), Some(&[0] as _), &[1, 2, 3]), + (take_in_bounds_range_from, (2..), Some(&[2, 3] as _), &[0, 1]), + (take_oob_range_to, (..5), None, &[0, 1, 2, 3]), + (take_oob_range_to_inclusive, (..=4), None, &[0, 1, 2, 3]), + (take_oob_range_from, (5..), None, &[0, 1, 2, 3]), +} + +take_tests! { + slice: &mut [0, 1, 2, 3], method: take_mut, + (take_mut_in_bounds_range_to, (..1), Some(&mut [0] as _), &mut [1, 2, 3]), + (take_mut_in_bounds_range_to_inclusive, (..=0), Some(&mut [0] as _), &mut [1, 2, 3]), + (take_mut_in_bounds_range_from, (2..), Some(&mut [2, 3] as _), &mut [0, 1]), + (take_mut_oob_range_to, (..5), None, &mut [0, 1, 2, 3]), + (take_mut_oob_range_to_inclusive, (..=4), None, &mut [0, 1, 2, 3]), + (take_mut_oob_range_from, (5..), None, &mut [0, 1, 2, 3]), +} + +take_tests! { + ty: &[_], slice: &[1, 2], method: take_first, + (take_first_nonempty, (), Some(&1), &[2]), +} + +take_tests! { + ty: &mut [_], slice: &mut [1, 2], method: take_first_mut, + (take_first_mut_nonempty, (), Some(&mut 1), &mut [2]), +} + +take_tests! { + ty: &[_], slice: &[1, 2], method: take_last, + (take_last_nonempty, (), Some(&2), &[1]), +} + +take_tests! { + ty: &mut [_], slice: &mut [1, 2], method: take_last_mut, + (take_last_mut_nonempty, (), Some(&mut 2), &mut [1]), +} + +take_tests! { + ty: &[()], slice: &[], method: take_first, + (take_first_empty, (), None, &[]), +} + +take_tests! { + ty: &mut [()], slice: &mut [], method: take_first_mut, + (take_first_mut_empty, (), None, &mut []), +} + +take_tests! { + ty: &[()], slice: &[], method: take_last, + (take_last_empty, (), None, &[]), +} + +take_tests! { + ty: &mut [()], slice: &mut [], method: take_last_mut, + (take_last_mut_empty, (), None, &mut []), +} + +const EMPTY_MAX: &'static [()] = &[(); std::usize::MAX]; + +// can't be a constant due to const mutability rules +// see https://github.com/rust-lang/rust/issues/57349#issuecomment-597395059 +macro_rules! empty_max_mut { + () => { + &mut [(); std::usize::MAX] as _ + }; +} + +take_tests! { + slice: &[(); ::std::usize::MAX], method: take, + (take_in_bounds_max_range_to, (..::std::usize::MAX), Some(EMPTY_MAX), &[(); 0]), + (take_oob_max_range_to_inclusive, (..=::std::usize::MAX), None, EMPTY_MAX), + (take_in_bounds_max_range_from, (::std::usize::MAX..), Some(&[] as _), EMPTY_MAX), +} + +take_tests! { + slice: &mut [(); ::std::usize::MAX], method: take_mut, + (take_mut_in_bounds_max_range_to, (..::std::usize::MAX), Some(empty_max_mut!()), &mut [(); 0]), + (take_mut_oob_max_range_to_inclusive, (..=::std::usize::MAX), None, empty_max_mut!()), + (take_mut_in_bounds_max_range_from, (::std::usize::MAX..), Some(&mut [] as _), empty_max_mut!()), +} From 23d74f29ff58816a7ebe51eaaf46311cb9ee72ab Mon Sep 17 00:00:00 2001 From: Tim Vermeulen Date: Tue, 22 Sep 2020 13:42:20 +0200 Subject: [PATCH 2/6] Remove hanging tests --- library/core/tests/slice.rs | 24 ------------------------ 1 file changed, 24 deletions(-) diff --git a/library/core/tests/slice.rs b/library/core/tests/slice.rs index 8f5386b2ba0e6..ac3c1e866e524 100644 --- a/library/core/tests/slice.rs +++ b/library/core/tests/slice.rs @@ -2118,27 +2118,3 @@ take_tests! { ty: &mut [()], slice: &mut [], method: take_last_mut, (take_last_mut_empty, (), None, &mut []), } - -const EMPTY_MAX: &'static [()] = &[(); std::usize::MAX]; - -// can't be a constant due to const mutability rules -// see https://github.com/rust-lang/rust/issues/57349#issuecomment-597395059 -macro_rules! empty_max_mut { - () => { - &mut [(); std::usize::MAX] as _ - }; -} - -take_tests! { - slice: &[(); ::std::usize::MAX], method: take, - (take_in_bounds_max_range_to, (..::std::usize::MAX), Some(EMPTY_MAX), &[(); 0]), - (take_oob_max_range_to_inclusive, (..=::std::usize::MAX), None, EMPTY_MAX), - (take_in_bounds_max_range_from, (::std::usize::MAX..), Some(&[] as _), EMPTY_MAX), -} - -take_tests! { - slice: &mut [(); ::std::usize::MAX], method: take_mut, - (take_mut_in_bounds_max_range_to, (..::std::usize::MAX), Some(empty_max_mut!()), &mut [(); 0]), - (take_mut_oob_max_range_to_inclusive, (..=::std::usize::MAX), None, empty_max_mut!()), - (take_mut_in_bounds_max_range_from, (::std::usize::MAX..), Some(&mut [] as _), empty_max_mut!()), -} From 5d6272ae60eebb3c3768cabeb851c898fb87bfa0 Mon Sep 17 00:00:00 2001 From: Tim Vermeulen Date: Tue, 22 Sep 2020 18:45:24 +0200 Subject: [PATCH 3/6] Adjust wording --- library/core/src/slice/mod.rs | 34 ++++++++++++++++------------------ 1 file changed, 16 insertions(+), 18 deletions(-) diff --git a/library/core/src/slice/mod.rs b/library/core/src/slice/mod.rs index 978c351d34276..bce0b5320c4b9 100644 --- a/library/core/src/slice/mod.rs +++ b/library/core/src/slice/mod.rs @@ -3188,10 +3188,11 @@ impl [T] { left } - /// Removes and returns the portion of the slice specified by `range`. + /// Returns the subslice corresponding to the given range, + /// and modifies the slice to no longer include this subslice. /// - /// If the provided `range` starts or ends outside of the slice, - /// `None` is returned and the slice is not modified. + /// Returns `None` and does not modify the slice if the given + /// range is out of bounds. /// /// # Examples /// @@ -3250,10 +3251,11 @@ impl [T] { } } - /// Removes and returns the portion of the mutable slice specified by `range`. + /// Returns the mutable subslice corresponding to the given range, + /// and modifies the slice to no longer include this subslice. /// - /// If the provided `range` starts or ends outside of the slice, - /// `None` is returned and the slice is not modified. + /// Returns `None` and does not modify the slice if the given + /// range is out of bounds. /// /// # Examples /// @@ -3315,9 +3317,8 @@ impl [T] { } } - /// Takes the first element out of the slice. - /// - /// Returns a reference pointing to the first element of the old slice. + /// Returns a reference to the first element of the slice, + /// and modifies the slice to no longer include this element. /// /// Returns `None` if the slice is empty. /// @@ -3338,9 +3339,8 @@ impl [T] { self.take(..=0).map(|res| &res[0]) } - /// Takes the first element out of the mutable slice. - /// - /// Returns a mutable reference pointing to the first element of the old slice. + /// Returns a mutable reference to the first element of the slice, + /// and modifies the slice to no longer include this element. /// /// Returns `None` if the slice is empty. /// @@ -3362,9 +3362,8 @@ impl [T] { self.take_mut(..=0).map(|res| &mut res[0]) } - /// Takes the last element out of the slice. - /// - /// Returns a reference pointing to the last element of the old slice. + /// Returns a reference to the last element of the slice, + /// and modifies the slice to no longer include this element. /// /// Returns `None` if the slice is empty. /// @@ -3385,9 +3384,8 @@ impl [T] { self.take((self.len() - 1)..).map(|res| &res[0]) } - /// Takes the last element out of the mutable slice. - /// - /// Returns a mutable reference pointing to the last element of the old slice. + /// Returns a mutable reference to the last element of the slice, + /// and modifies the slice to no longer include this element. /// /// Returns `None` if the slice is empty. /// From 8dda5c4fc477e9738d177dff709377c78b01aae0 Mon Sep 17 00:00:00 2001 From: Tim Vermeulen Date: Tue, 22 Sep 2020 19:02:58 +0200 Subject: [PATCH 4/6] Avoid underflow --- library/core/src/slice/mod.rs | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/library/core/src/slice/mod.rs b/library/core/src/slice/mod.rs index bce0b5320c4b9..aaf30aa0d257f 100644 --- a/library/core/src/slice/mod.rs +++ b/library/core/src/slice/mod.rs @@ -3381,7 +3381,8 @@ impl [T] { #[inline] #[unstable(feature = "slice_take", issue = "62280")] pub fn take_last<'a>(self: &mut &'a Self) -> Option<&'a T> { - self.take((self.len() - 1)..).map(|res| &res[0]) + let i = self.len().checked_sub(1)?; + self.take(i..).map(|res| &res[0]) } /// Returns a mutable reference to the last element of the slice, @@ -3404,7 +3405,8 @@ impl [T] { #[inline] #[unstable(feature = "slice_take", issue = "62280")] pub fn take_last_mut<'a>(self: &mut &'a mut Self) -> Option<&'a mut T> { - self.take_mut((self.len() - 1)..).map(|res| &mut res[0]) + let i = self.len().checked_sub(1)?; + self.take_mut(i..).map(|res| &mut res[0]) } } From e35ba521472995dce3cd926d170567c112a62e1a Mon Sep 17 00:00:00 2001 From: Tim Vermeulen Date: Thu, 24 Sep 2020 02:57:11 +0200 Subject: [PATCH 5/6] Don't replace `self` by empty slice in `slice::take` --- library/core/src/slice/mod.rs | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/library/core/src/slice/mod.rs b/library/core/src/slice/mod.rs index aaf30aa0d257f..26360dedcb65d 100644 --- a/library/core/src/slice/mod.rs +++ b/library/core/src/slice/mod.rs @@ -3240,8 +3240,7 @@ impl [T] { if split_index > self.len() { return None; } - let original = crate::mem::take(self); - let (front, back) = original.split_at(split_index); + let (front, back) = self.split_at(split_index); if taking_front { *self = back; Some(front) From 81b8edc9b42a3c05ccbeadbc3e28389dac23b6e7 Mon Sep 17 00:00:00 2001 From: Tim Vermeulen Date: Wed, 21 Oct 2020 08:52:54 +0200 Subject: [PATCH 6/6] Implement `take_first` methods in terms of `split_first` methods --- library/core/src/slice/mod.rs | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/library/core/src/slice/mod.rs b/library/core/src/slice/mod.rs index 26360dedcb65d..ef936af229a82 100644 --- a/library/core/src/slice/mod.rs +++ b/library/core/src/slice/mod.rs @@ -3335,7 +3335,9 @@ impl [T] { #[inline] #[unstable(feature = "slice_take", issue = "62280")] pub fn take_first<'a>(self: &mut &'a Self) -> Option<&'a T> { - self.take(..=0).map(|res| &res[0]) + let (first, rem) = self.split_first()?; + *self = rem; + Some(first) } /// Returns a mutable reference to the first element of the slice, @@ -3358,7 +3360,9 @@ impl [T] { #[inline] #[unstable(feature = "slice_take", issue = "62280")] pub fn take_first_mut<'a>(self: &mut &'a mut Self) -> Option<&'a mut T> { - self.take_mut(..=0).map(|res| &mut res[0]) + let (first, rem) = mem::take(self).split_first_mut()?; + *self = rem; + Some(first) } /// Returns a reference to the last element of the slice, @@ -3380,8 +3384,9 @@ impl [T] { #[inline] #[unstable(feature = "slice_take", issue = "62280")] pub fn take_last<'a>(self: &mut &'a Self) -> Option<&'a T> { - let i = self.len().checked_sub(1)?; - self.take(i..).map(|res| &res[0]) + let (last, rem) = self.split_last()?; + *self = rem; + Some(last) } /// Returns a mutable reference to the last element of the slice, @@ -3404,8 +3409,9 @@ impl [T] { #[inline] #[unstable(feature = "slice_take", issue = "62280")] pub fn take_last_mut<'a>(self: &mut &'a mut Self) -> Option<&'a mut T> { - let i = self.len().checked_sub(1)?; - self.take_mut(i..).map(|res| &mut res[0]) + let (last, rem) = mem::take(self).split_last_mut()?; + *self = rem; + Some(last) } }