Skip to content

Commit ba6436f

Browse files
authored
Rollup merge of rust-lang#62421 - JohnTitor:U007D-master, r=alexcrichton
Introduce `as_deref` to Option This is re-submission for rust-lang#59628. Renames `deref()` to `as_deref()` and adds `deref_mut()` impls and tests. CC rust-lang#50264 r? @Kimundi (I picked you as you're the previous reviewer.)
2 parents b7b2a4c + 59634bc commit ba6436f

31 files changed

+393
-136
lines changed

src/libcore/option.rs

+14-3
Original file line numberDiff line numberDiff line change
@@ -136,7 +136,7 @@
136136
#![stable(feature = "rust1", since = "1.0.0")]
137137

138138
use crate::iter::{FromIterator, FusedIterator, TrustedLen};
139-
use crate::{convert, fmt, hint, mem, ops::{self, Deref}};
139+
use crate::{convert, fmt, hint, mem, ops::{self, Deref, DerefMut}};
140140
use crate::pin::Pin;
141141

142142
// Note that this is not a lang item per se, but it has a hidden dependency on
@@ -1104,17 +1104,28 @@ impl<T: Default> Option<T> {
11041104

11051105
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
11061106
impl<T: Deref> Option<T> {
1107-
/// Converts from `&Option<T>` to `Option<&T::Target>`.
1107+
/// Converts from `Option<T>` (or `&Option<T>`) to `Option<&T::Target>`.
11081108
///
11091109
/// Leaves the original Option in-place, creating a new one with a reference
11101110
/// to the original one, additionally coercing the contents via [`Deref`].
11111111
///
11121112
/// [`Deref`]: ../../std/ops/trait.Deref.html
1113-
pub fn deref(&self) -> Option<&T::Target> {
1113+
pub fn as_deref(&self) -> Option<&T::Target> {
11141114
self.as_ref().map(|t| t.deref())
11151115
}
11161116
}
11171117

1118+
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
1119+
impl<T: DerefMut> Option<T> {
1120+
/// Converts from `Option<T>` (or `&mut Option<T>`) to `Option<&mut T::Target>`.
1121+
///
1122+
/// Leaves the original `Option` in-place, creating a new one containing a mutable reference to
1123+
/// the inner type's `Deref::Target` type.
1124+
pub fn as_deref_mut(&mut self) -> Option<&mut T::Target> {
1125+
self.as_mut().map(|t| t.deref_mut())
1126+
}
1127+
}
1128+
11181129
impl<T, E> Option<Result<T, E>> {
11191130
/// Transposes an `Option` of a [`Result`] into a [`Result`] of an `Option`.
11201131
///

src/libcore/result.rs

+49-16
Original file line numberDiff line numberDiff line change
@@ -232,7 +232,7 @@
232232

233233
use crate::fmt;
234234
use crate::iter::{FromIterator, FusedIterator, TrustedLen};
235-
use crate::ops::{self, Deref};
235+
use crate::ops::{self, Deref, DerefMut};
236236

237237
/// `Result` is a type that represents either success ([`Ok`]) or failure ([`Err`]).
238238
///
@@ -981,42 +981,75 @@ impl<T: Default, E> Result<T, E> {
981981

982982
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
983983
impl<T: Deref, E> Result<T, E> {
984-
/// Converts from `&Result<T, E>` to `Result<&T::Target, &E>`.
984+
/// Converts from `Result<T, E>` (or `&Result<T, E>`) to `Result<&T::Target, &E>`.
985985
///
986-
/// Leaves the original Result in-place, creating a new one with a reference
987-
/// to the original one, additionally coercing the `Ok` arm of the Result via
988-
/// `Deref`.
989-
pub fn deref_ok(&self) -> Result<&T::Target, &E> {
986+
/// Leaves the original `Result` in-place, creating a new one containing a reference to the
987+
/// `Ok` type's `Deref::Target` type.
988+
pub fn as_deref_ok(&self) -> Result<&T::Target, &E> {
990989
self.as_ref().map(|t| t.deref())
991990
}
992991
}
993992

994993
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
995994
impl<T, E: Deref> Result<T, E> {
996-
/// Converts from `&Result<T, E>` to `Result<&T, &E::Target>`.
995+
/// Converts from `Result<T, E>` (or `&Result<T, E>`) to `Result<&T, &E::Target>`.
997996
///
998-
/// Leaves the original Result in-place, creating a new one with a reference
999-
/// to the original one, additionally coercing the `Err` arm of the Result via
1000-
/// `Deref`.
1001-
pub fn deref_err(&self) -> Result<&T, &E::Target>
997+
/// Leaves the original `Result` in-place, creating a new one containing a reference to the
998+
/// `Err` type's `Deref::Target` type.
999+
pub fn as_deref_err(&self) -> Result<&T, &E::Target>
10021000
{
10031001
self.as_ref().map_err(|e| e.deref())
10041002
}
10051003
}
10061004

10071005
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
10081006
impl<T: Deref, E: Deref> Result<T, E> {
1009-
/// Converts from `&Result<T, E>` to `Result<&T::Target, &E::Target>`.
1007+
/// Converts from `Result<T, E>` (or `&Result<T, E>`) to `Result<&T::Target, &E::Target>`.
10101008
///
1011-
/// Leaves the original Result in-place, creating a new one with a reference
1012-
/// to the original one, additionally coercing both the `Ok` and `Err` arms
1013-
/// of the Result via `Deref`.
1014-
pub fn deref(&self) -> Result<&T::Target, &E::Target>
1009+
/// Leaves the original `Result` in-place, creating a new one containing a reference to both
1010+
/// the `Ok` and `Err` types' `Deref::Target` types.
1011+
pub fn as_deref(&self) -> Result<&T::Target, &E::Target>
10151012
{
10161013
self.as_ref().map(|t| t.deref()).map_err(|e| e.deref())
10171014
}
10181015
}
10191016

1017+
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
1018+
impl<T: DerefMut, E> Result<T, E> {
1019+
/// Converts from `Result<T, E>` (or `&mut Result<T, E>`) to `Result<&mut T::Target, &mut E>`.
1020+
///
1021+
/// Leaves the original `Result` in-place, creating a new one containing a mutable reference to
1022+
/// the `Ok` type's `Deref::Target` type.
1023+
pub fn as_deref_mut_ok(&mut self) -> Result<&mut T::Target, &mut E> {
1024+
self.as_mut().map(|t| t.deref_mut())
1025+
}
1026+
}
1027+
1028+
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
1029+
impl<T, E: DerefMut> Result<T, E> {
1030+
/// Converts from `Result<T, E>` (or `&mut Result<T, E>`) to `Result<&mut T, &mut E::Target>`.
1031+
///
1032+
/// Leaves the original `Result` in-place, creating a new one containing a mutable reference to
1033+
/// the `Err` type's `Deref::Target` type.
1034+
pub fn as_deref_mut_err(&mut self) -> Result<&mut T, &mut E::Target>
1035+
{
1036+
self.as_mut().map_err(|e| e.deref_mut())
1037+
}
1038+
}
1039+
1040+
#[unstable(feature = "inner_deref", reason = "newly added", issue = "50264")]
1041+
impl<T: DerefMut, E: DerefMut> Result<T, E> {
1042+
/// Converts from `Result<T, E>` (or `&mut Result<T, E>`) to
1043+
/// `Result<&mut T::Target, &mut E::Target>`.
1044+
///
1045+
/// Leaves the original `Result` in-place, creating a new one containing a mutable reference to
1046+
/// both the `Ok` and `Err` types' `Deref::Target` types.
1047+
pub fn as_deref_mut(&mut self) -> Result<&mut T::Target, &mut E::Target>
1048+
{
1049+
self.as_mut().map(|t| t.deref_mut()).map_err(|e| e.deref_mut())
1050+
}
1051+
}
1052+
10201053
impl<T, E> Result<Option<T>, E> {
10211054
/// Transposes a `Result` of an `Option` into an `Option` of a `Result`.
10221055
///

src/libcore/tests/option.rs

+25-5
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
use core::option::*;
22
use core::mem;
33
use core::clone::Clone;
4+
use core::array::FixedSizeArray;
5+
use core::ops::DerefMut;
46

57
#[test]
68
fn test_get_ptr() {
@@ -310,20 +312,38 @@ fn test_try() {
310312
}
311313

312314
#[test]
313-
fn test_option_deref() {
315+
fn test_option_as_deref() {
314316
// Some: &Option<T: Deref>::Some(T) -> Option<&T::Deref::Target>::Some(&*T)
315317
let ref_option = &Some(&42);
316-
assert_eq!(ref_option.deref(), Some(&42));
318+
assert_eq!(ref_option.as_deref(), Some(&42));
317319

318320
let ref_option = &Some(String::from("a result"));
319-
assert_eq!(ref_option.deref(), Some("a result"));
321+
assert_eq!(ref_option.as_deref(), Some("a result"));
320322

321323
let ref_option = &Some(vec![1, 2, 3, 4, 5]);
322-
assert_eq!(ref_option.deref(), Some(&[1, 2, 3, 4, 5][..]));
324+
assert_eq!(ref_option.as_deref(), Some([1, 2, 3, 4, 5].as_slice()));
323325

324326
// None: &Option<T: Deref>>::None -> None
325327
let ref_option: &Option<&i32> = &None;
326-
assert_eq!(ref_option.deref(), None);
328+
assert_eq!(ref_option.as_deref(), None);
329+
}
330+
331+
#[test]
332+
fn test_option_as_deref_mut() {
333+
// Some: &mut Option<T: Deref>::Some(T) -> Option<&mut T::Deref::Target>::Some(&mut *T)
334+
let mut val = 42;
335+
let ref_option = &mut Some(&mut val);
336+
assert_eq!(ref_option.as_deref_mut(), Some(&mut 42));
337+
338+
let ref_option = &mut Some(String::from("a result"));
339+
assert_eq!(ref_option.as_deref_mut(), Some(String::from("a result").deref_mut()));
340+
341+
let ref_option = &mut Some(vec![1, 2, 3, 4, 5]);
342+
assert_eq!(ref_option.as_deref_mut(), Some([1, 2, 3, 4, 5].as_mut_slice()));
343+
344+
// None: &mut Option<T: Deref>>::None -> None
345+
let ref_option: &mut Option<&mut i32> = &mut None;
346+
assert_eq!(ref_option.as_deref_mut(), None);
327347
}
328348

329349
#[test]

0 commit comments

Comments
 (0)