Skip to content

Commit ec48989

Browse files
authored
Rollup merge of #73577 - VillSnow:master, r=Amanieu
Add partition_point Add partition_point in C++. Although existing binary_search in rust does not suitable when the slice has multiple hits, this function returns exact point of partition. The definition of this function is very clear and able to accept general matter, therefore you can easily get index which you want like lower/upper_bound. rust-lang/rfcs#2184
2 parents 25687ca + 6f8ad3b commit ec48989

File tree

3 files changed

+95
-0
lines changed

3 files changed

+95
-0
lines changed

src/libcore/slice/mod.rs

+54
Original file line numberDiff line numberDiff line change
@@ -2663,6 +2663,60 @@ impl<T> [T] {
26632663
{
26642664
self.iter().is_sorted_by_key(f)
26652665
}
2666+
2667+
/// Returns the index of the partition point according to the given predicate
2668+
/// (the index of the first element of the second partition).
2669+
///
2670+
/// The slice is assumed to be partitioned according to the given predicate.
2671+
/// This means that all elements for which the predicate returns true are at the start of the slice
2672+
/// and all elements for which the predicate returns false are at the end.
2673+
/// For example, [7, 15, 3, 5, 4, 12, 6] is a partitioned under the predicate x % 2 != 0
2674+
/// (all odd numbers are at the start, all even at the end).
2675+
///
2676+
/// If this slice is not partitioned, the returned result is unspecified and meaningless,
2677+
/// as this method performs a kind of binary search.
2678+
///
2679+
/// # Examples
2680+
///
2681+
/// ```
2682+
/// #![feature(partition_point)]
2683+
///
2684+
/// let v = [1, 2, 3, 3, 5, 6, 7];
2685+
/// let i = v.partition_point(|&x| x < 5);
2686+
///
2687+
/// assert_eq!(i, 4);
2688+
/// assert!(v[..i].iter().all(|&x| x < 5));
2689+
/// assert!(v[i..].iter().all(|&x| !(x < 5)));
2690+
/// ```
2691+
#[unstable(feature = "partition_point", reason = "new API", issue = "73831")]
2692+
pub fn partition_point<P>(&self, mut pred: P) -> usize
2693+
where
2694+
P: FnMut(&T) -> bool,
2695+
{
2696+
let mut left = 0;
2697+
let mut right = self.len();
2698+
2699+
while left != right {
2700+
let mid = left + (right - left) / 2;
2701+
// SAFETY:
2702+
// When left < right, left <= mid < right.
2703+
// Therefore left always increases and right always decreases,
2704+
// and eigher of them is selected.
2705+
// In both cases left <= right is satisfied.
2706+
// Therefore if left < right in a step,
2707+
// left <= right is satisfied in the next step.
2708+
// Therefore as long as left != right, 0 <= left < right <= len is satisfied
2709+
// and if this case 0 <= mid < len is satisfied too.
2710+
let value = unsafe { self.get_unchecked(mid) };
2711+
if pred(value) {
2712+
left = mid + 1;
2713+
} else {
2714+
right = mid;
2715+
}
2716+
}
2717+
2718+
left
2719+
}
26662720
}
26672721

26682722
#[lang = "slice_u8"]

src/libcore/tests/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@
4444
#![feature(const_forget)]
4545
#![feature(option_unwrap_none)]
4646
#![feature(peekable_next_if)]
47+
#![feature(partition_point)]
4748

4849
extern crate test;
4950

src/libcore/tests/slice.rs

+40
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,46 @@ fn test_binary_search_implementation_details() {
8181
assert_eq!(b.binary_search(&3), Ok(8));
8282
}
8383

84+
#[test]
85+
fn test_partition_point() {
86+
let b: [i32; 0] = [];
87+
assert_eq!(b.partition_point(|&x| x < 5), 0);
88+
89+
let b = [4];
90+
assert_eq!(b.partition_point(|&x| x < 3), 0);
91+
assert_eq!(b.partition_point(|&x| x < 4), 0);
92+
assert_eq!(b.partition_point(|&x| x < 5), 1);
93+
94+
let b = [1, 2, 4, 6, 8, 9];
95+
assert_eq!(b.partition_point(|&x| x < 5), 3);
96+
assert_eq!(b.partition_point(|&x| x < 6), 3);
97+
assert_eq!(b.partition_point(|&x| x < 7), 4);
98+
assert_eq!(b.partition_point(|&x| x < 8), 4);
99+
100+
let b = [1, 2, 4, 5, 6, 8];
101+
assert_eq!(b.partition_point(|&x| x < 9), 6);
102+
103+
let b = [1, 2, 4, 6, 7, 8, 9];
104+
assert_eq!(b.partition_point(|&x| x < 6), 3);
105+
assert_eq!(b.partition_point(|&x| x < 5), 3);
106+
assert_eq!(b.partition_point(|&x| x < 8), 5);
107+
108+
let b = [1, 2, 4, 5, 6, 8, 9];
109+
assert_eq!(b.partition_point(|&x| x < 7), 5);
110+
assert_eq!(b.partition_point(|&x| x < 0), 0);
111+
112+
let b = [1, 3, 3, 3, 7];
113+
assert_eq!(b.partition_point(|&x| x < 0), 0);
114+
assert_eq!(b.partition_point(|&x| x < 1), 0);
115+
assert_eq!(b.partition_point(|&x| x < 2), 1);
116+
assert_eq!(b.partition_point(|&x| x < 3), 1);
117+
assert_eq!(b.partition_point(|&x| x < 4), 4);
118+
assert_eq!(b.partition_point(|&x| x < 5), 4);
119+
assert_eq!(b.partition_point(|&x| x < 6), 4);
120+
assert_eq!(b.partition_point(|&x| x < 7), 4);
121+
assert_eq!(b.partition_point(|&x| x < 8), 5);
122+
}
123+
84124
#[test]
85125
fn test_iterator_nth() {
86126
let v: &[_] = &[0, 1, 2, 3, 4];

0 commit comments

Comments
 (0)