课程名称:领略Rust之美,挑战双高语言
课程章节: 第3章 Rust进阶
主讲老师:叶枭
课程内容:
今天学习的内容是 Rust 中所有权的转移。
还是以这段代码为例:
fn main() {
let s1 = String::from("hello");
let s2 = s1;
println!("{}", s1);
}
这段代码运行会报错,提示变量 s1 的值已经被移动了。
对于 JSer 来说,这有些不易理解。因为如果是按照 JS 中的语法来看,let s2 = s1
就是做了一个赋值操作,将变量 s1 的值赋值给了变量 s2,s1 是可以正常打印的。
而在 Rust 中,这里有两点是和 JS 不同的。一是字符串值属于复杂数据类型,存在堆内存中。二是由于所有权的存在,导致一个值只能有一个所有者。所以在将 s1 的值交给 s2 之后,值的所有者变为了 s2,s1 不再有对值的控制权,所以再打印 s1 就会报错。
这个过程涉及到了一个新的特性,所有权的转移。
所有权的转移
Rust 中的复杂数据类型的值,都存在堆内存中。比如:
let s1 = String::from("hello");
let s2 = s1;
第一行代码运行时,会申请一块堆内存空间来存放字符串值,然后将该堆内存的指针地址,赋值给变量 s1,存在栈内存中。
第二行代码,是将变量 s1 所绑定的一个指针地址,转移给了变量 s2,导致 s1 不再拥有对指针地址的所有权。
这个过程就叫所有权的转移。
上面的代码示例中是通过赋值操作,将值的所有权进行了转译。除此之外,函数传参,函数返回等操作也会发生所有权的转移,比如:
fn main() {
let s1 = String::from("hello");
// 将变量 s1 对字符串值的所有权转移给了函数 reverse。到这一步,s1 就失去了对值的所有权。
// 函数 reverse 执行完成,s2 拥有了函数返回值的所有权
let s2 = reverse(s1);
println!("{}", s2);
}
// 参数 s 拿到了字符串值的所有权,在函数作用域内可以使用该值
fn reverse(s: String) -> String {
// 函数通过返回值,又将 s 的所有权转移回去
s.chars().rev().collect()
}
课程收获
这节课学习了所有权的转移,对于一个复杂数据类型的值,在执行赋值操作,函数传参,函数返回等操作时,会将所有权进行转移。
点击查看更多内容
为 TA 点赞
评论
共同学习,写下你的评论
评论加载中...
作者其他优质文章
正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦