// Copyright 2013 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. use std::vec; use std::util; /// A vector type optimized for cases where the size is almost always 0 or 1 pub enum SmallVector { priv Zero, priv One(T), priv Many(~[T]), } impl Container for SmallVector { fn len(&self) -> uint { match *self { Zero => 0, One(..) => 1, Many(ref vals) => vals.len() } } } impl FromIterator for SmallVector { fn from_iterator>(iter: &mut I) -> SmallVector { let mut v = Zero; for val in *iter { v.push(val); } v } } impl SmallVector { pub fn zero() -> SmallVector { Zero } pub fn one(v: T) -> SmallVector { One(v) } pub fn many(vs: ~[T]) -> SmallVector { Many(vs) } pub fn push(&mut self, v: T) { match *self { Zero => *self = One(v), One(..) => { let one = util::replace(self, Zero); match one { One(v1) => util::replace(self, Many(~[v1, v])), _ => unreachable!() }; } Many(ref mut vs) => vs.push(v) } } pub fn get<'a>(&'a self, idx: uint) -> &'a T { match *self { One(ref v) if idx == 0 => v, Many(ref vs) => &vs[idx], _ => fail!("out of bounds access") } } pub fn expect_one(self, err: &'static str) -> T { match self { One(v) => v, Many([v]) => v, _ => fail!(err) } } pub fn move_iter(self) -> MoveItems { match self { Zero => ZeroIterator, One(v) => OneIterator(v), Many(vs) => ManyIterator(vs.move_iter()) } } } pub enum MoveItems { priv ZeroIterator, priv OneIterator(T), priv ManyIterator(vec::MoveItems), } impl Iterator for MoveItems { fn next(&mut self) -> Option { match *self { ZeroIterator => None, OneIterator(..) => { let mut replacement = ZeroIterator; util::swap(self, &mut replacement); match replacement { OneIterator(v) => Some(v), _ => unreachable!() } } ManyIterator(ref mut inner) => inner.next() } } fn size_hint(&self) -> (uint, Option) { match *self { ZeroIterator => (0, Some(0)), OneIterator(..) => (1, Some(1)), ManyIterator(ref inner) => inner.size_hint() } } } #[cfg(test)] mod test { use super::*; #[test] fn test_len() { let v: SmallVector = SmallVector::zero(); assert_eq!(0, v.len()); assert_eq!(1, SmallVector::one(1).len()); assert_eq!(5, SmallVector::many(~[1, 2, 3, 4, 5]).len()); } #[test] fn test_push_get() { let mut v = SmallVector::zero(); v.push(1); assert_eq!(1, v.len()); assert_eq!(&1, v.get(0)); v.push(2); assert_eq!(2, v.len()); assert_eq!(&2, v.get(1)); v.push(3); assert_eq!(3, v.len()); assert_eq!(&3, v.get(2)); } #[test] fn test_from_iterator() { let v: SmallVector = (~[1, 2, 3]).move_iter().collect(); assert_eq!(3, v.len()); assert_eq!(&1, v.get(0)); assert_eq!(&2, v.get(1)); assert_eq!(&3, v.get(2)); } #[test] fn test_move_iter() { let v = SmallVector::zero(); let v: ~[int] = v.move_iter().collect(); assert_eq!(~[], v); let v = SmallVector::one(1); assert_eq!(~[1], v.move_iter().collect()); let v = SmallVector::many(~[1, 2, 3]); assert_eq!(~[1, 2, 3], v.move_iter().collect()); } #[test] #[should_fail] fn test_expect_one_zero() { let _: int = SmallVector::zero().expect_one(""); } #[test] #[should_fail] fn test_expect_one_many() { SmallVector::many(~[1, 2]).expect_one(""); } #[test] fn test_expect_one_one() { assert_eq!(1, SmallVector::one(1).expect_one("")); assert_eq!(1, SmallVector::many(~[1]).expect_one("")); } }