Skip to content

Commit 515a18c

Browse files
authored
Merge pull request #187 from conradludgate/into-inner-take
take and into_inner_unchecked functions
2 parents dfceb6d + 0fdf39f commit 515a18c

File tree

2 files changed

+49
-5
lines changed

2 files changed

+49
-5
lines changed

src/arrayvec.rs

+25-5
Original file line numberDiff line numberDiff line change
@@ -636,14 +636,34 @@ impl<T, const CAP: usize> ArrayVec<T, CAP> {
636636
if self.len() < self.capacity() {
637637
Err(self)
638638
} else {
639-
unsafe {
640-
let self_ = ManuallyDrop::new(self);
641-
let array = ptr::read(self_.as_ptr() as *const [T; CAP]);
642-
Ok(array)
643-
}
639+
unsafe { Ok(self.into_inner_unchecked()) }
644640
}
645641
}
646642

643+
/// Return the inner fixed size array.
644+
///
645+
/// Safety:
646+
/// This operation is safe if and only if length equals capacity.
647+
pub unsafe fn into_inner_unchecked(self) -> [T; CAP] {
648+
debug_assert_eq!(self.len(), self.capacity());
649+
let self_ = ManuallyDrop::new(self);
650+
let array = ptr::read(self_.as_ptr() as *const [T; CAP]);
651+
array
652+
}
653+
654+
/// Returns the ArrayVec, replacing the original with a new empty ArrayVec.
655+
///
656+
/// ```
657+
/// use arrayvec::ArrayVec;
658+
///
659+
/// let mut v = ArrayVec::from([0, 1, 2, 3]);
660+
/// assert_eq!([0, 1, 2, 3], v.take().into_inner().unwrap());
661+
/// assert!(v.is_empty());
662+
/// ```
663+
pub fn take(&mut self) -> Self {
664+
mem::replace(self, Self::new())
665+
}
666+
647667
/// Return a slice containing all elements of the vector.
648668
pub fn as_slice(&self) -> &[T] {
649669
ArrayVecImpl::as_slice(self)

tests/tests.rs

+24
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,21 @@ fn test_drop() {
153153
assert_eq!(flag.get(), 3);
154154
}
155155

156+
// test take
157+
flag.set(0);
158+
{
159+
let mut array1 = ArrayVec::<_, 3>::new();
160+
array1.push(Bump(flag));
161+
array1.push(Bump(flag));
162+
array1.push(Bump(flag));
163+
let array2 = array1.take();
164+
assert_eq!(flag.get(), 0);
165+
drop(array1);
166+
assert_eq!(flag.get(), 0);
167+
drop(array2);
168+
assert_eq!(flag.get(), 3);
169+
}
170+
156171
// test cloning into_iter
157172
flag.set(0);
158173
{
@@ -461,6 +476,15 @@ fn test_into_inner_3() {
461476
assert_eq!(v.into_inner().unwrap(), [1, 2, 3, 4]);
462477
}
463478

479+
#[test]
480+
fn test_take() {
481+
let mut v1 = ArrayVec::<i32, 4>::new();
482+
v1.extend(1..=4);
483+
let v2 = v1.take();
484+
assert!(v1.into_inner().is_err());
485+
assert_eq!(v2.into_inner().unwrap(), [1, 2, 3, 4]);
486+
}
487+
464488
#[cfg(feature="std")]
465489
#[test]
466490
fn test_write() {

0 commit comments

Comments
 (0)