2016-06-29 285 views
0

我试图在对它进行一些操作后返回一个可变引用。这是最好的一段代码解释:在Rust中借用并返回可变引用的优雅方法

#[derive(PartialEq)] 
pub enum Value { 
    Null, 
    Array(Vec<Value>), 
} 

impl Value { 
    pub fn new() -> Value { 
     Value::Array(Vec::new()) 
    } 

    pub fn push<'a, T> (&'a mut self, value: T) -> Option<&'a mut Value> 
    where T:Into<Value> { 
     let temp = match *self { 
      Value::Array(ref mut vec) => { 
       vec.push(value.into()); 
       true 
      }, 
      _ => false, 
     }; 
     if temp { 
      Some(self) 
     } else { 
      None 
     } 
    } 
} 

#[test] 
fn push_test() { 
    let mut val = Value::new(); 
    val.push(Value::Null); 
    assert!(val == Value::Array(vec![Value::Null])); 
} 

的打版是here。布尔值的解决方法是因为如果我从match块中返回Some(self),我将多次借用。有没有一种优雅的方式来实现push函数而不使用布尔值?如果有可能保留功能签名,那么它是一个奖金。谢谢!

回答

1

与布尔值的解决方法是,因为我会借多次,如果我回到Some(self)从匹配块

另一种选择是内更换self时间,所以v可以采取的所有权矢量(避免借用)。在将新项目添加到v后,我们重建self值:

// the lifetime 'a can be omitted 
pub fn push<T>(&mut self, value: T) -> Option<&mut Value> 
    where T: Into<Value> 
{ 
    // replace put Value::Null on self and return the old value 
    match ::std::mem::replace(self, Value::Null) { 
     Value::Array(mut v) => { 
      v.push(value.into()); 
      *self = Value::Array(v); 
      Some(self) 
     }, 
     _ => None, 
    } 
}