rust - 如何在 StackVec 上实现 IntoIterator

标签 rust iterator lifetime

我正在尝试编写一个类似 Vec 的具有固定大小的对象,以便在堆栈上使用。

这是我要完成的在线类(class)的一部分 Assignment 1, Phase 2A .

我在实现 IntoIterator 特性时遇到问题。

StructVec

的代码
#![no_std]

pub struct StackVec<'a, T: 'a> {
    storage: &'a mut [T],
    len: usize
}

impl<'a, T: 'a> StackVec<'a, T> {
    pub fn new(storage: &'a mut [T]) -> StackVec<'a, T> {
        StackVec { storage, len: 0 }
    }

    pub fn with_len(storage: &'a mut [T], len: usize) -> StackVec<'a, T> {
        StackVec { storage, len }
    }

    pub fn capacity(&self) -> usize {
        self.storage.len()
    }

    pub fn truncate(&mut self, len: usize) {
        if len < self.len {
            self.len = len;
        }
    }

    pub fn into_slice(self) -> &'a mut [T] {
        &mut self.storage[..self.len]
    }

    pub fn as_slice(&self) -> &[T] {
        &self.storage[..self.len]
    }

    pub fn as_mut_slice(&mut self) -> &mut [T] {
        &mut self.storage[..self.len]
    }

    pub fn len(&self) -> usize {
        self.len
    }

    pub fn is_empty(&self) -> bool {
        self.len == 0
    }

    pub fn is_full(&self) -> bool {
        self.len == self.storage.len()
    }

    pub fn push(&mut self, value: T) -> Result<(), ()> {
        if self.is_full() {
            return Err(());
        }
        self.storage[self.len] = value;
        self.len += 1;
        Ok(())
    }
}

我能够通过返回底层数组的迭代器来实现 IntoIterator 特性:

impl<'a, T: 'a> IntoIterator for StackVec<'a, T> {
    type Item = &'a mut T;
    type IntoIter = core::slice::IterMut<'a, T>;

    fn into_iter(self) -> Self::IntoIter {
        self.storage.into_iter()
    }
}

然而,这并不是我真正想要的,因为它将遍历整个数组,而不仅仅是被推到它上面的项目。

我已经尝试从底层数组的数组切片返回一个迭代器,但我被生命周期困住了:

impl<'a, T: 'a> IntoIterator for StackVec<'a, T> {
    type Item = &'a T;
    type IntoIter = core::slice::Iter<'a, T>;

    fn into_iter(self) -> Self::IntoIter {
        self.as_slice().into_iter()
    }
}

编译失败

error[E0597]: `self` does not live long enough
   --> src/lib.rs:165:9
    |
165 |         self.as_slice().into_iter()
    |         ^^^^ borrowed value does not live long enough
166 |     }
    |     - borrowed value only lives until here
    |
note: borrowed value must be valid for the lifetime 'a as defined on the impl at 160:1...
   --> src/lib.rs:160:1
    |
160 | / impl<'a, T: 'a> IntoIterator for StackVec<'a, T> {
161 | |     type Item = &'a T;
162 | |     type IntoIter = core::slice::Iter<'a, T>;
163 | |
...   |
166 | |     }
167 | | }
    | |_^

我也尝试过完全创建一个新的 Iterator 但我又被抓了!

struct Iter<'a, T: 'a> {
    stack_vec: StackVec<'a, T>,
    start: usize,
}

impl<'a, T: 'a> Iterator for Iter<'a, T> {
    type Item = T;

    fn next(&mut self) -> Option<Self::Item> {
        let start = self.start;
        if start < self.stack_vec.len {
            self.start += 1;
            Some(self.stack_vec[start])
        } else {
            None
        }
    }

    fn size_hint(&self) -> (usize, Option<usize>) {
        let size = self.stack_vec.len - self.start;
        (size, Some(size))
    }
}

编译失败,错误:

error[E0508]: cannot move out of type `[T]`, a non-copy slice
   --> src/lib.rs:148:18
    |
148 |             Some(self.stack_vec[start])
    |                  ^^^^^^^^^^^^^^^^^^^^^ cannot move out of here

我已经尝试查看做同样事情的 crate ,但要么他们使用我不能用于此作业的 std 库,要么我不明白如何移植他们的代码到我的。

https://github.com/bluss/arrayvec

https://github.com/danielhenrymantilla/stackvec-rs/blob/master/src/stackvec/traits/into_iter.rs

我做错了什么?

最佳答案

作为您最初想法的一个相当简单的扩展,您可以使用 take 方法限制迭代器可以返回的元素数量。这会将您的 IntoIterator 实现变成:


impl<'a, T: 'a> IntoIterator for StackVec<'a, T> {
    type Item = &'a mut T;
    type IntoIter = std::iter::Take<core::slice::IterMut<'a, T>>;

    fn into_iter(self) -> Self::IntoIter {
        self.storage.into_iter().take(self.len)
    }
}

您在尝试实现 IntoIterator 时遇到了问题,因为 into_iter 方法获取了 self 的所有权。这意味着一旦函数退出,您需要确保您已经从 self 中移动或复制了您想要在迭代器中使用的所有内容。由于 as_slice 仅借用,因此您无法从函数中将迭代器返回到切片(它借用的数据将被删除)。

impl<'a, T: 'a> IntoIterator for StackVec<'a, T> {
    type Item = &'a T;
    type IntoIter = core::slice::Iter<'a, T>;

    fn into_iter(self) -> Self::IntoIter {
        //       ^ no & or &mut here means this function takes ownership of self
        self.as_slice().into_iter()
    }
}

您仍然可以为您的 StackVec 结构创建一个非拥有迭代器,但您不会使用 IntoIterator 来完成它。相反,只需在名为 iter 的结构上创建一个方法:

pub fn iter(&self) -> impl Iterator<Item=&T> {
    self.storage.iter().take(self.len)
}

关于rust - 如何在 StackVec 上实现 IntoIterator,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58358439/

相关文章:

pattern-matching - Rust: "cannot move out of ` self` 因为它是借来的”错误

rust - 如何注释此 Rust/Calloop 回调代码的生命周期?

rust - 为什么元组或结构的大小不是成员的总和?

java - ListIterator 以前的方法不起作用

reference - 有没有办法返回对函数中创建的变量的引用?

rust - 在 Clap 的一个参数中取多个值

c++ - 条件中的迭代器赋值 - vector 迭代器不兼容

c++ - 迭代 vector <int> 或数值范围时如何避免代码重复?

rust - 对指针和它指向的结构的生命周期参数使用相同的生命周期

c++ - 从 C++ 中的析构函数中恢复对象?