js做示爱网站例子,河北新闻最新消息今天,深圳进出口贸易有限公司,珠海网站建设科速互联内部可变性#xff08;interior mutability#xff09;是RUST的设计模式之一#xff0c;它允许你在只持有不可变引用的前提下对数据进行修改。为了能改变数据#xff0c;内部可变性模式在它的数据结构中使用了unsafe#xff08;不安全#xff09;代码来绕过RUST正常的可变…内部可变性interior mutability是RUST的设计模式之一它允许你在只持有不可变引用的前提下对数据进行修改。为了能改变数据内部可变性模式在它的数据结构中使用了unsafe不安全代码来绕过RUST正常的可变性和借用规则。
假如我们能够保证自己的代码在运行时符合借用规则那么即使编译器无法在编译阶段保证符合借用规则也能使用那些采用了内部可变性模式的类型。
对于一般引用和BoxT的代码RUST会在编译阶段强制代码遵守借用规则。而对于RefCellT的代码RUST则只会在运行时检查这些规则并在违反借用规则的情况下触发panic来终止程序。
与RcT相似RefCellT只能被用于单线程场景中。强行将它用于多线程环境中会产生编译时错误。下面是选择使用BoxT、RcT和RefCellT的依据
RcT允许一份数据有多个所有者而BoxT和RefCellT都只有一个所有者BoxT允许在编译时检查可变或不可变借用RcT仅允许编译时检查不可变借用RefCellT允许运行时检查可变或不可变借用。由于RefCellT允许我们在运行时检查可变借用所以即使RefCellT本身是不可变的我们仍然能够更改其中存储的值。
pub trait Messager {fn send(self, msg: str);
}pub struct LimitTrackera, T: a Messager {messager: a T,value: usize,
}impla, T LimitTrackera, T
whereT: Messager,
{pub fn new(messager: T, value: usize) - LimitTrackerT {LimitTracker {messager: messager,value: value,}}pub fn set_value(mut self, value: usize) {self.value value;self.messager.send(msg);}
}这段代码的重点是Messager trait它唯一的send方法可以接受self的不可变引用及一条文本消息作为参数。
示例的结构体同时使用生命周期标注和泛型约束结构体实现关联函数new和方法set_value。紧跟impl后的尖括号声明指明结构体自身的约束。
下面对set_value做单元测试mock一个实现了Message trait的结构体MockMessage进行验证调用send方法会向结构体字段中写入值。
#[cfg(test)]
mod test {use super::*;struct MockMessage {sent_message: VecString,}impl MockMessage {fn new() - MockMessage {MockMessage {sent_message: vec![],}}}impl Messager for MockMessage {fn send(self, msg: str) {self.sent_message.push(String::from(msg));}}#[test]fn send_warning_message() {let mock_messager MockMessage::new();let mut limit_tracker LimitTracker::new(mock_messager, 2);limit_tracker.set_value(10);}
}编译器报错send方法接收的self属于不可变引用我们无法修改MockMessage中的内容。如果将send方法self修改为mut self又不符合Messager trait约束。
在保持外部值不可变的前提下使用RefCellT来修改内部存储的值。sent_message字段类型调整成RefCellVecString最后调整成下面这个样子
#[cfg(test)]
mod test {use super::*;use std::cell::RefCell;struct MockMessage {sent_message: RefCellVecString,}impl MockMessage {fn new() - MockMessage {MockMessage {sent_message: RefCell::new(vec![]),}}}impl Messager for MockMessage {fn send(self, msg: str) {self.sent_message.borrow_mut().push(String::from(msg));}}#[test]fn send_warning_message() {let mock_messager MockMessage::new();let mut limit_tracker LimitTracker::new(mock_messager, 2);limit_tracker.set_value(10);}
}我们调用RefCellVecString类型的borrow_mut方法来获取内部值的可变引用接着我们便可以在动态数据的可变引用上调用push方法来存入数据。
我们会在创建不可变引用和可变引用时分别使用语法和mut。RefCellT实现了borrow和borrow_mut方法分别返回RefT与RefMutT这两种智能指针。由于这两种指针都实现了Deref所以我们可以把它们当做一般的引用来对待。
RefCellT会记录当前存在多少个活跃的RefT和RefMutT智能指针在任何一个给定的时间里它只允许你拥有多个不可变借用和一个可变借用。当我们违背借用规则时相比于一般引用导致的编译错误RefCellT会在运行时触发panic。