blob: 74e48f2b2c8c993d4f7595132b50b0b67923dcca [file] [log] [blame]
mod linear_group;
mod linear_group_by;
mod linear_group_by_key;
pub use self::linear_group::{LinearGroup, LinearGroupMut};
pub use self::linear_group_by::{LinearGroupBy, LinearGroupByMut};
pub use self::linear_group_by_key::{LinearGroupByKey, LinearGroupByKeyMut};
#[cfg(test)]
mod tests {
use super::*;
#[derive(Debug, Eq)]
enum Guard {
Valid(i32),
Invalid(i32),
}
impl PartialEq for Guard {
fn eq(&self, other: &Self) -> bool {
match (self, other) {
(Guard::Valid(_), Guard::Valid(_)) => true,
(a, b) => panic!("denied read on Guard::Invalid variant ({:?}, {:?})", a, b),
}
}
}
#[test]
fn one_big_group() {
let slice = &[1, 1, 1, 1];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next(), Some(&[1, 1, 1, 1][..]));
assert_eq!(iter.next(), None);
}
#[test]
fn two_equal_groups() {
let slice = &[1, 1, 1, 1, 2, 2, 2, 2];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next(), Some(&[1, 1, 1, 1][..]));
assert_eq!(iter.next(), Some(&[2, 2, 2, 2][..]));
assert_eq!(iter.next(), None);
}
#[test]
fn two_little_equal_groups() {
let slice = &[1, 2];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next(), Some(&[1][..]));
assert_eq!(iter.next(), Some(&[2][..]));
assert_eq!(iter.next(), None);
}
#[test]
fn three_groups() {
let slice = &[1, 1, 1, 3, 3, 2, 2, 2];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next(), Some(&[1, 1, 1][..]));
assert_eq!(iter.next(), Some(&[3, 3][..]));
assert_eq!(iter.next(), Some(&[2, 2, 2][..]));
assert_eq!(iter.next(), None);
}
#[test]
fn three_little_groups() {
let slice = &[1, 3, 2];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next(), Some(&[1][..]));
assert_eq!(iter.next(), Some(&[3][..]));
assert_eq!(iter.next(), Some(&[2][..]));
assert_eq!(iter.next(), None);
}
#[test]
fn overflow() {
let slice = &[Guard::Invalid(0), Guard::Valid(1), Guard::Valid(2), Guard::Invalid(3)];
let mut iter = LinearGroup::new(&slice[1..3]);
assert_eq!(iter.next(), Some(&[Guard::Valid(1), Guard::Valid(2)][..]));
assert_eq!(iter.next(), None);
}
#[test]
fn last_three_little_groups() {
let slice = &[1, 3, 2];
let iter = LinearGroup::new(slice);
assert_eq!(iter.last(), Some(&[2][..]));
}
#[test]
fn last_three_groups() {
let slice = &[1, 1, 1, 3, 3, 2, 2, 2];
let iter = LinearGroup::new(slice);
assert_eq!(iter.last(), Some(&[2, 2, 2][..]));
}
#[test]
fn last_overflow() {
let slice = &[Guard::Invalid(0), Guard::Valid(1), Guard::Valid(2), Guard::Invalid(3)];
println!("{:?}", (&slice[1..3]).as_ptr());
let iter = LinearGroup::new(&slice[1..3]);
assert_eq!(iter.last(), Some(&[Guard::Valid(1), Guard::Valid(2)][..]));
}
#[test]
fn back_empty_slice() {
let slice: &[i32] = &[];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next_back(), None);
}
#[test]
fn back_one_little_group() {
let slice = &[1];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next_back(), Some(&[1][..]));
assert_eq!(iter.next_back(), None);
assert_eq!(iter.next(), None);
}
#[test]
fn back_three_little_groups() {
let slice = &[1, 3, 2];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next_back(), Some(&[2][..]));
assert_eq!(iter.next_back(), Some(&[3][..]));
assert_eq!(iter.next_back(), Some(&[1][..]));
assert_eq!(iter.next_back(), None);
}
#[test]
fn back_three_groups() {
let slice = &[1, 1, 1, 3, 3, 2, 2, 2];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next_back(), Some(&[2, 2, 2][..]));
assert_eq!(iter.next_back(), Some(&[3, 3][..]));
assert_eq!(iter.next_back(), Some(&[1, 1, 1][..]));
assert_eq!(iter.next_back(), None);
}
#[test]
fn double_ended_dont_cross() {
let slice = &[1, 1, 1, 3, 3, 2, 2, 2];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next(), Some(&[1, 1, 1][..]));
assert_eq!(iter.next_back(), Some(&[2, 2, 2][..]));
assert_eq!(iter.next(), Some(&[3, 3][..]));
assert_eq!(iter.next_back(), None);
assert_eq!(iter.next(), None);
}
#[test]
fn fused_iterator() {
let slice = &[1, 2, 3];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next(), Some(&[1][..]));
assert_eq!(iter.next(), Some(&[2][..]));
assert_eq!(iter.next(), Some(&[3][..]));
assert_eq!(iter.next(), None);
assert_eq!(iter.next(), None);
}
#[test]
fn back_fused_iterator() {
let slice = &[1, 2, 3];
let mut iter = LinearGroup::new(slice);
assert_eq!(iter.next_back(), Some(&[3][..]));
assert_eq!(iter.next_back(), Some(&[2][..]));
assert_eq!(iter.next_back(), Some(&[1][..]));
assert_eq!(iter.next_back(), None);
assert_eq!(iter.next_back(), None);
}
fn panic_param_ord(a: &i32, b: &i32) -> bool {
if a < b { true }
else { panic!("params are not in the right order") }
}
#[test]
fn predicate_call_param_order() {
let slice = &[1, 2, 3, 4, 5];
let mut iter = LinearGroupBy::new(slice, panic_param_ord);
assert_eq!(iter.next(), Some(&[1, 2, 3, 4, 5][..]));
assert_eq!(iter.next(), None);
}
#[test]
fn rev_predicate_call_param_order() {
let slice = &[1, 2, 3, 4, 5];
let mut iter = LinearGroupBy::new(slice, panic_param_ord);
assert_eq!(iter.next_back(), Some(&[1, 2, 3, 4, 5][..]));
assert_eq!(iter.next_back(), None);
}
#[test]
fn group_by_key_mut() {
let slice = &mut [1, 2, 4, 5, 7, 8, 8];
let mut iter = LinearGroupByKeyMut::new(slice, |i: &i32| *i % 2);
assert_eq!(iter.next(), Some(&mut [1][..]));
assert_eq!(iter.next(), Some(&mut [2, 4][..]));
assert_eq!(iter.next(), Some(&mut [5, 7][..]));
assert_eq!(iter.next(), Some(&mut [8, 8][..]));
assert_eq!(iter.next(), None);
}
}
#[cfg(all(feature = "nightly", test))]
mod bench {
extern crate test;
extern crate rand;
use super::*;
use self::rand::{Rng, SeedableRng};
use self::rand::rngs::StdRng;
use self::rand::distributions::Alphanumeric;
#[bench]
fn vector_16_000(b: &mut test::Bencher) {
let mut rng = StdRng::from_seed([42; 32]);
let len = 16_000;
let mut vec = Vec::with_capacity(len);
for _ in 0..len {
vec.push(rng.sample(Alphanumeric));
}
b.iter(|| {
let group_by = LinearGroup::new(vec.as_slice());
test::black_box(group_by.count())
})
}
#[bench]
fn vector_16_000_sorted(b: &mut test::Bencher) {
let mut rng = StdRng::from_seed([42; 32]);
let len = 16_000;
let mut vec = Vec::with_capacity(len);
for _ in 0..len {
vec.push(rng.sample(Alphanumeric));
}
vec.sort_unstable();
b.iter(|| {
let group_by = LinearGroup::new(vec.as_slice());
test::black_box(group_by.count())
})
}
#[bench]
fn vector_little_sorted(b: &mut test::Bencher) {
let mut rng = StdRng::from_seed([42; 32]);
let len = 30;
let mut vec = Vec::with_capacity(len);
for _ in 0..len {
vec.push(rng.sample(Alphanumeric));
}
vec.sort_unstable();
b.iter(|| {
let group_by = LinearGroup::new(vec.as_slice());
test::black_box(group_by.count())
})
}
#[bench]
fn vector_16_000_one_group(b: &mut test::Bencher) {
let vec = vec![1; 16_000];
b.iter(|| {
let group_by = LinearGroup::new(vec.as_slice());
test::black_box(group_by.count())
})
}
#[bench]
fn rev_vector_16_000(b: &mut test::Bencher) {
let mut rng = StdRng::from_seed([42; 32]);
let len = 16_000;
let mut vec = Vec::with_capacity(len);
for _ in 0..len {
vec.push(rng.sample(Alphanumeric));
}
b.iter(|| {
let group_by = LinearGroup::new(vec.as_slice());
test::black_box(group_by.rev().count())
})
}
#[bench]
fn rev_vector_16_000_one_group(b: &mut test::Bencher) {
let vec = vec![1; 16_000];
b.iter(|| {
let group_by = LinearGroup::new(vec.as_slice());
test::black_box(group_by.rev().count())
})
}
}