我在对象中有2个向量(Result
s)。第一个向量all_results
拥有其内容,第二个向量current_results
包含引用
到all_results
的内容。这是代码
struct Result {
val: int
}
struct Obj<'a> {
all_results: Vec<Result>,
current_results: Vec<&'a mut Result>
}
impl<'a> Obj<'a> {
fn change_something (&'a mut self) -> &[&'a mut Result] {
let mut newVec: Vec<&'a mut Result> = Vec::new();
for item in self.current_results.mut_iter() {
// type of item is &mut &'a mut Result
item.val += 1;
//need to create valid reference here
if (item.val == 1) {
newVec.push(*item); //dereferenced item is now &'a mut Result
}
}
self.current_results = newVec;
self.current_results.as_slice()
}
}
fn main () {
let a = Result { val: 0 };
let b = Result { val: 1 };
let mut obj = Obj { all_results: vec![], current_results: vec![] };
obj.all_results.push(a);
obj.all_results.push(b);
obj.change_something();
}
基本上我尝试做的是浏览current_results
中的每个条目,修改它们指向的Result
,然后根据某些条件过滤条目。
但是,我无法使用相同的引用,因为编译器抱怨item
的生存时间长于其范围允许的时间。
iter_lifetime2.rs:16:29: 16:34 error: lifetime of `item` is too short to guarantee its contents can be safely reborrowed
iter_lifetime2.rs:16 newVec.push(*item);
^~~~~
iter_lifetime2.rs:11:61: 21:6 note: `item` would have to be valid for the lifetime 'a as defined on the block at 11:60...
iter_lifetime2.rs:11 fn change_something (&'a mut self) -> &[&'a mut Result] {
iter_lifetime2.rs:12 let mut newVec: Vec<&'a mut Result> = Vec::new();
iter_lifetime2.rs:13 for item in self.current_results.mut_iter() {
iter_lifetime2.rs:14 item.val += 1;
iter_lifetime2.rs:15 if (item.val == 1) {
iter_lifetime2.rs:16 newVec.push(*item);
...
iter_lifetime2.rs:13:9: 19:13 note: ...but `item` is only valid for the expression at 13:8
iter_lifetime2.rs:13 for item in self.current_results.mut_iter() {
iter_lifetime2.rs:14 item.val += 1;
iter_lifetime2.rs:15 if (item.val == 1) {
iter_lifetime2.rs:16 newVec.push(*item);
iter_lifetime2.rs:17 }
iter_lifetime2.rs:18 }
我尝试了几种变体,例如:
for item in self.current_results.mut_iter() {
let p: &'a mut Result = *item;
item.val += 1;
if (item.val == 1) {
newVec.push(p);
}
}
但是我在item
这是我的向量的基本指针图,希望这使我的意图更清晰。谢谢!
问题:
有没有办法从现有参考创建参考?感觉我应该能够生成'a
生命周期的另一个指针并使其指向'a
生命周期的对象。或者有更好的方法来解决这样的问题吗?
答案 0 :(得分:1)
Vec
有一个retain
方法几乎可以满足您的需求。在过滤之前,它不允许元素的变异。我们可以编写自己的retain_mut
方法。我刚刚复制了不可变版本的代码并对其进行了修改以允许变异。
trait RetainMut<T> {
fn retain_mut(&mut self, f: |&mut T| -> bool);
}
impl<T> RetainMut<T> for Vec<T> {
fn retain_mut(&mut self, f: |&mut T| -> bool) {
let len = self.len();
let mut del = 0u;
{
let v = self.as_mut_slice();
for i in range(0u, len) {
if !f(&mut v[i]) {
del += 1;
} else if del > 0 {
v.swap(i-del, i);
}
}
}
if del > 0 {
self.truncate(len - del);
}
}
}
然后你的代码看起来像这样
struct Result {
val: int
}
struct Obj<'a> {
all_results: Vec<Result>,
current_results: Vec<&'a mut Result>
}
impl<'a> Obj<'a> {
fn change_something (&mut self) -> &[&mut Result] {
self.current_results.retain_mut(|result| {
result.val += 1;
result.val == 1
});
self.current_results.as_slice()
}
}