rust 引用包括&
和&mut
,引用即允许你使用值但是不获取所有权,所以当引用离开其作用域之后其指向的值也不会丢弃。
&
:也可以称为借用(borrowing),是没有权利变更引用数据内容的&mut
:称为可变引用,是可以变更引用数据内容的。
let s1 = String::from("hellow");
let s = &s1;//s 是s1的引用
其内存结构如图:
引用使用限制
- 在特定作用域要么有且只有一个可变引用,要么只能有多个不可变引用
- 引用在特定作用域内必须总是有效
特定作用域:指从应用声明的地方开始一直持续到
最后一次使用
为止。
实例1:不可变引用特定作用域多次引用(OK
)
let mut guess = String::from("hellow");
//特定作用域拥有多个不可变应用OK
let r2 = &guess;
let r1 = &guess;
println!("{}--{}",r1,r2);
实例2:可变引用特定作用域有且只允许有一个引用(ERR
)
let mut guess = String::from("hellow");
//特定作用域拥有且只允许有一个可变引用
let r2 = &mut guess;
let r1 = &guess;//由于存在r2可变应用的时候,不在允许存在其他的引用,所以这里error
let r3 = &mut guess;//由于存在r2可变应用的时候,不在允许存在其他的引用,所以这里error
println!("{}--{}",r1,r2);//引用作用域结束
实例3:可变引用特定作用之外多次引用(OK
)
let mut guess = String::from("hellow");
let r2 = &mut guess;//r2作用域开始
println!("{}",r2);//r2作用域结束
let r1 = &guess;//由于r2作用域已经结束,所以r1引用合理
println!("{}",r1);
垂悬引用
垂悬引用:即其指向的内存可能已经被释放或者分配给其他持有者。rust在编译的时候会报错。
fun dangle()-> &String{
let s = String::from("test");
&s;//这里由于s已经离开作用域,所以返回的引用区域内存已经被释放,从而产生垂悬引用
}
Rust支持自动引用和解引用
由于rust不支持->
,所以p->xxx
是不被允许的,我们只能通过解引用之后使用xxx,(*p).xxx
。但是由于Rust支持自动解引用,所以
(*p).xxx
和p.xxx
等价。
//这两行代码是等价的
p.xxx;
(*p).xxx;