当前位置: 首页 > news >正文

学习笔记九——Rust所有权机制

🦀 Rust 所有权机制

📚 目录

  1. 什么是值类型和引用类型?值语义和引用语义?
  2. 什么是所有权?为什么 Rust 需要它?
  3. 所有权的三大原则(修正版)
  4. 移动语义 vs 复制语义:变量赋值到底发生了什么?
  5. 绑定和作用域:值的“活多久”与“谁管它”
  6. 引用和借用:本质是什么?为什么这么设计?
  7. 借用的三条规则(逐条解释+示例)
  8. 生命周期:引用能用多久?
  9. 生命周期参数:为什么需要?怎么用?
  10. Rust 的借用检查器是怎么帮你“兜底”的?
  11. 总结与学习建议

1️⃣ 什么是值类型和引用类型?值语义和引用语义?

✔️ 值类型(Value Type)

变量里存的是实际的数据。

let x = 10;  // x 是 i32 值类型

✔️ 引用类型(Reference Type)

变量里存的是指向其他数据的“地址”。

let x = 10;
let y = &x;  // y 是对 x 的引用,类型是 &i32

🔍 值语义 vs 引用语义

概念解释示例
值语义每次赋值都复制/移动数据本身let b = a;
引用语义多个变量共享同一个值的访问权(指针)let r = &a;

2️⃣ 什么是所有权?为什么 Rust 需要它?

在 Rust 中,所有权机制用于管理内存,确保内存安全,不靠垃圾回收(GC),也不手动 free

通俗理解:谁拥有这个数据,谁负责清理它。

  • 避免悬空引用
  • 自动释放资源(变量离开作用域)
  • 避免多线程数据竞争

3️⃣ Rust 所有权的三大原则

  1. 每个值在任意时刻只能有一个所有者。
  2. 当所有者离开作用域,值会被自动释放。
  3. 一个值只能有一个可变引用,或任意多个不可变引用,但不能同时存在。
let a = String::from("hello");
let b = a;        // a 的所有权移动到 b
// println!("{}", a); // ❌ 报错:a 被移动了

4️⃣ 移动语义 vs 复制语义

let a = 10;        // a 是 Copy 类型
let b = a;         // 自动复制,a 仍然有效

let s1 = String::from("Hi");
let s2 = s1;       // 所有权移动,s1 失效

想要深拷贝,用 .clone()

let s1 = String::from("Rust");
let s2 = s1.clone(); // ✅ s1 仍有效

5️⃣ 绑定和作用域

变量只在其“作用域”内有效:

{
    let name = String::from("Tom");
    println!("{}", name);
} // name 离开作用域,自动释放

作用域由 {} 包围,函数、if、match、循环等都会创建作用域。


6️⃣ 引用和借用:到底有什么区别?

✅ 本质解释:

  • 引用(reference)是“变量的地址”(通过 & 获取)
  • 借用(borrowing)是“使用别人的值但不拥有它”的行为

🧠 所以:“引用”是工具,而“借用”是动作

举例说明:

let s = String::from("Rust");
let r = &s; // 你用 &s 创建了一个“引用”,这就是“借用了 s 的值”
概念通俗解释Rust 表现形式
引用地址&a&mut a
借用临时借东西不拥有let r = &a

7️⃣ 借用的三条规则(逐条解释+示例)

规则 1:借用不能比原值活得更久(生命周期限制)

fn dangle() -> &String {
    let s = String::from("oops");
    &s // ❌ s 被释放,引用悬空
}

规则 2:可变借用具有独占性

let mut s = String::from("hi");
let r1 = &mut s;
// let r2 = &mut s; // ❌ 同时只能有一个可变借用

规则 3:有不可变借用时,不能再借出可变引用

let mut s = String::from("hello");
let r1 = &s;
let r2 = &s;
// let r3 = &mut s; // ❌ 报错:不能混合借用

📌 原则:只读可以多个,可写只能一个,不能混用。


8️⃣ 生命周期:引用能用多久?

生命周期就是引用在内存中能活多久

Rust 不允许引用指向已释放的内存:

fn dangle() -> &String {
    let s = String::from("hi");
    &s // ❌ 返回了已经释放的 s
}

9️⃣ 生命周期参数:为什么需要?怎么用?

当函数的参数和返回值中有引用,Rust 要你说明它们“谁活得久”。这就是“生命周期参数”。

fn longest<'a>(x: &'a str, y: &'a str) -> &'a str {
    if x.len() > y.len() { x } else { y }
}

解释:

  • <'a> 是生命周期参数
  • xy、返回值 都必须活得跟 'a 一样久

📌 一般来说,编译器可以自动推断。但在函数返回引用时,必须显式写出生命周期参数。


🔟 Rust 的借用检查器是怎么帮你的?

Rust 有一个非常强大的“借用检查器”(Borrow Checker),会在编译阶段检查:

  • 有没有悬空引用?
  • 可变引用有没有被别名?
  • 是否混用了可变和不可变引用?
  • 生命周期是否符合规则?

🧠 你写错了引用规则,Rust 不会让你运行,而是直接编译失败。

✅ 借用检查器就是 Rust 安全的“守门员”!它会严格替你把关内存安全。


🔚 总结与学习建议

概念通俗解释
所有权谁拥有值,谁负责清理
移动所有权转移,原值失效
复制创建副本,两个变量互不影响
借用临时访问但不拥有,分可变/不可变
引用表示地址,借用的语法工具
生命周期引用能活多久(不能活得比原值久)
生命周期参数显式告诉编译器“谁和谁一样久”

相关文章:

  • 探索 MCP 和 A2A 协议: 本质上新协议都基于 HTTP的
  • 【Spring底层分析】Spring IoC
  • RocketMQ 安装时启动一闪而过 start mqnamesrv.cmd
  • 插值算法 - 最近邻插值实现
  • 用python写一个简单的射击游戏
  • skynet.socket 完整通信流程
  • Mysql8配置文件
  • 【贪心之摆动序列】
  • 三、Virtual Device Manager
  • CST1019.基于Spring Boot+Vue智能洗车管理系统
  • 从零手写RPC-version0
  • Android游戏逆向工程全面指南
  • MySQL慢查询全攻略:定位、分析与优化实战
  • Python 深度学习 第5章 机器学习的核心问题泛化及如何提高模型的泛化能力实例
  • 【MySQL】关于何时使用start slave和start slave user=‘’ password=‘’
  • 计算机的发展及应用
  • minio重大更新!RELEASE.2025-04-08T15-41-24Z发布:修复关键Bug,存储管理更高效!
  • 基于PyQt5的Jupyter Notebook转Python工具
  • 生物化学笔记:医学免疫学原理16 自身免疫和自身免疫性疾病
  • 电感特性参数、选型方法与厂商推荐
  • AI创业者聊大模型应用趋势:可用性和用户需求是关键
  • 石家庄桥西区通报“中药液”添加安眠药问题:对医院立案调查
  • 梅花奖在上海|秦海璐:演了15年《四世同堂》,想演一辈子
  • 回家了!子弹库帛书二、三卷将于7月首次面向公众展出
  • 中国驻美大使:远离故土的子弹库帛书正随民族复兴踏上归途
  • 时隔3年,持续近2小时,俄乌在土耳其谈成了什么?