Skip to content

Commit f3eceba

Browse files
committed
slice: allow [T]::contains to accept any Q where T: Borrow<Q>
This is most useful to allow matching string literals against a `[String]`, or bytestring literals against a `[Vec<u8>]`. Since `Vec<T>` derefs to `[T]`, this also applies to `Vec<String>` and `Vec<Vec<u8>>`.
1 parent dc2003b commit f3eceba

File tree

4 files changed

+32
-6
lines changed

4 files changed

+32
-6
lines changed

src/liballoc/slice.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -973,8 +973,9 @@ impl<T> [T] {
973973
/// assert!(!v.contains(&50));
974974
/// ```
975975
#[stable(feature = "rust1", since = "1.0.0")]
976-
pub fn contains(&self, x: &T) -> bool
977-
where T: PartialEq
976+
pub fn contains<Q: ?Sized>(&self, x: &Q) -> bool
977+
where T: Borrow<Q>,
978+
Q: PartialEq
978979
{
979980
core_slice::SliceExt::contains(self, x)
980981
}

src/liballoc/tests/slice.rs

+20
Original file line numberDiff line numberDiff line change
@@ -367,6 +367,26 @@ fn test_binary_search() {
367367
assert_eq!([1, 2, 3, 4, 5].binary_search(&0).ok(), None);
368368
}
369369

370+
#[test]
371+
fn test_contains() {
372+
assert!([5, 4, 3, 2, 1].contains(&2));
373+
let empty: &[i32] = &[][..];
374+
assert!(!empty.contains(&1));
375+
376+
// Tests that Borrow functions.
377+
let string_slice = [String::from("abc"), String::from("def")];
378+
assert!(string_slice.contains(&String::from("abc")));
379+
assert!(!string_slice.contains(&String::from("ab")));
380+
assert!(string_slice.contains("abc"));
381+
assert!(!string_slice.contains("ab"));
382+
383+
let vecu8_slice = [b"abc".to_vec(), b"def".to_vec()];
384+
assert!(vecu8_slice.contains(&b"abc".to_vec()));
385+
assert!(!vecu8_slice.contains(&b"ab".to_vec()));
386+
assert!(vecu8_slice.contains(&b"abc"[..]));
387+
assert!(!vecu8_slice.contains(&b"ab"[..]));
388+
}
389+
370390
#[test]
371391
fn test_reverse() {
372392
let mut v = vec![10, 20];

src/libcore/slice/mod.rs

+8-3
Original file line numberDiff line numberDiff line change
@@ -195,7 +195,9 @@ pub trait SliceExt {
195195
fn as_mut_ptr(&mut self) -> *mut Self::Item;
196196

197197
#[stable(feature = "core", since = "1.6.0")]
198-
fn contains(&self, x: &Self::Item) -> bool where Self::Item: PartialEq;
198+
fn contains<Q: ?Sized>(&self, x: &Q) -> bool
199+
where Self::Item: Borrow<Q>,
200+
Q: PartialEq;
199201

200202
#[stable(feature = "core", since = "1.6.0")]
201203
fn starts_with(&self, needle: &[Self::Item]) -> bool where Self::Item: PartialEq;
@@ -616,8 +618,11 @@ impl<T> SliceExt for [T] {
616618
}
617619

618620
#[inline]
619-
fn contains(&self, x: &T) -> bool where T: PartialEq {
620-
self.iter().any(|elt| *x == *elt)
621+
fn contains<Q: ?Sized>(&self, x: &Q) -> bool
622+
where T: Borrow<Q>,
623+
Q: PartialEq
624+
{
625+
self.iter().any(|elt| *x == *elt.borrow())
621626
}
622627

623628
#[inline]

src/librustc/ty/inhabitedness/def_id_forest.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -110,7 +110,7 @@ impl<'a, 'gcx, 'tcx> DefIdForest {
110110
where I: IntoIterator<Item=DefIdForest>
111111
{
112112
let mut ret = DefIdForest::empty();
113-
let mut next_ret = SmallVec::new();
113+
let mut next_ret: SmallVec<[DefId; 1]> = SmallVec::new();
114114
for next_forest in iter {
115115
for id in ret.root_ids.drain(..) {
116116
if !next_forest.contains(tcx, id) {

0 commit comments

Comments
 (0)