rust学习内存安全探秘:变量的所有权引用与借用

author author     2023-02-10     459

关键词:

作者:京东零售 周凯

一.前言

Rust 语言由 Mozilla 开发,最早发布于 2014 年 9 月,是一种高效、可靠的通用高级语言。其高效不仅限于开发效率,它的执行效率也是令人称赞的,是一种少有的兼顾开发效率和执行效率的语言。Rust语言具备如下特性:

•高性能 - Rust 速度惊人且内存利用率极高。由于没有运行时和垃圾回收,它能够胜任对性能要求特别高的服务,可以在嵌入式设备上运行,还能轻松和其他语言集成。

•可靠性 - Rust 丰富的类型系统和所有权模型保证了内存安全和线程安全,让您在编译期就能够消除各种各样的错误。

•生产力 - Rust 拥有出色的文档、友好的编译器和清晰的错误提示信息, 还集成了一流的工具 —— 包管理器和构建工具, 智能地自动补全和类型检验的多编辑器支持, 以及自动格式化代码等等。

Rust最近几年发展非常迅速,广受一线程序员的欢迎,Rust有一个官方维护的模块库(​​crates.io: Rust Package Registry​​),可以通过编译器自带的cargo管理工具方便的引入模块,目前crates.io上面的模块数量已经突破10万个,仍在快速增长,此情此景仿佛过去10年node.js的发展情景再现。

12月11日,Linus Torvalds发布了Linux6.1内核稳定版,并带来一个重磅的新闻,即Linux6.1将包含对Rust语言的原生支持。尽管这一功能仍在构建中,不过这也意味着,在可见的将来,Linux的历史将翻开崭新的一页——除了C之外,开发人员将第一次能够使用另一种语言Rust进行内核开发。

在近几年的讨论中,是否在Linux内核中引入Rust多次成为议题。不过包括 Torvalds在内的一众关键人物均对此表示了期待。早在2019年,Alex Gaynor和Geoffrey Thomas就曾于Linux Security Summit安全峰会上进行了演讲。他们指出,在Android和Ubuntu中,约有三分之二的内核漏洞被分配到CVE中,这些漏洞都是来自于内存安全问题。原则上,Rust可以通过其type system和borrow checker所提供的更安全的API来完全避免这类错误。简言之,Rust比C更安全。谷歌Android团队的Wedson Almeida Filho也曾公开表示:“我们觉得Rust现在已经准备好加入C语言,作为实现内核的实用语言。它可以帮助我们减少特权代码中潜在错误和安全漏洞的数量,同时很好地与核心内核配合并保留其性能特征。”

当前,谷歌在Android中广泛使用Rust。在那里,“目标不是将现有的C/C++转换为Rust,而是随着时间的推移,将新代码的开发转移到内存安全语言”。这一言论也逐渐在实践中得到论证。“随着进入Android的新内存不安全代码的数量减少,内存安全漏洞的数量也在减少。从2019年到2022年,相关漏洞占比已从Android总漏洞的76%下降到35%。2022年,在Android漏洞排行中,内存安全漏洞第一次不再是主因。”

本文将探寻相比于其他语言,Rust是怎样实现内存安全的。Rust针对创建于内存堆上的复杂数据类型,设计了一套独有的内存管理机制,该套机制包含变量的所有权机制、变量的作用域、变量的引用与借用,并专门针对字符串、数组、元组等复杂类型设计了slice类型,下面将具体讲述这些机制与规则。

二.变量的所有权

Rust 的核心功能(之一)是 所有权ownership)。虽然该功能很容易解释,但它对语言的其他部分有着深刻的影响。

所有程序都必须管理其运行时使用计算机内存的方式。一些语言中具有垃圾回收机制,在程序运行时有规律地寻找不再使用的内存;在另一些语言中,程序员必须亲自分配和释放内存。Rust 则选择了第三种方式:通过所有权系统管理内存,编译器在编译时会根据一系列的规则进行检查。如果违反了任何这些规则,程序都不能编译。在运行时,所有权系统的任何功能都不会减慢程序。

因为所有权对很多程序员来说都是一个新概念,需要一些时间来适应。好消息是随着你对 Rust 和所有权系统的规则越来越有经验,你就越能自然地编写出安全和高效的代码。持之以恒!

当你理解了所有权,你将有一个坚实的基础来理解那些使 Rust 独特的功能。在本章中,我们将通过完成一些示例来介绍所有权,这些示例基于一个常用的数据结构:字符串。

栈(Stack)与堆(Heap)在很多语言中,你并不需要经常考虑到栈与堆。不过在像 Rust 这样的系统编程语言中,值是位于栈上还是堆上在更大程度上影响了语言的行为以及为何必须做出这样的抉择。我们会在本文的稍后部分描述所有权与栈和堆相关的内容,所以这里只是一个用来预热的简要解释。栈和堆都是代码在运行时可供使用的内存,但是它们的结构不同。栈以放入值的顺序存储值并以相反顺序取出值。这也被称作 后进先出(last in, first out)。想象一下一叠盘子:当增加更多盘子时,把它们放在盘子堆的顶部,当需要盘子时,也从顶部拿走。不能从中间也不能从底部增加或拿走盘子!增加数据叫做 进栈(pushing onto the stack),而移出数据叫做 出栈(popping off the stack)。栈中的所有数据都必须占用已知且固定的大小。在编译时大小未知或大小可能变化的数据,要改为存储在堆上。 堆是缺乏组织的:当向堆放入数据时,你要请求一定大小的空间。内存分配器(memory allocator)在堆的某处找到一块足够大的空位,把它标记为已使用,并返回一个表示该位置地址的 指针(pointer)。这个过程称作 在堆上分配内存(allocating on the heap),有时简称为 “分配”(allocating)。(将数据推入栈中并不被认为是分配)。因为指向放入堆中数据的指针是已知的并且大小是固定的,你可以将该指针存储在栈上,不过当需要实际数据时,必须访问指针。想象一下去餐馆就座吃饭。当进入时,你说明有几个人,餐馆员工会找到一个够大的空桌子并领你们过去。如果有人来迟了,他们也可以通过询问来找到你们坐在哪。入栈比在堆上分配内存要快,因为(入栈时)分配器无需为存储新数据去搜索内存空间;其位置总是在栈顶。相比之下,在堆上分配内存则需要更多的工作,这是因为分配器必须首先找到一块足够存放数据的内存空间,并接着做一些记录为下一次分配做准备。访问堆上的数据比访问栈上的数据慢,因为必须通过指针来访问。现代处理器在内存中跳转越少就越快(缓存)。继续类比,假设有一个服务员在餐厅里处理多个桌子的点菜。在一个桌子报完所有菜后再移动到下一个桌子是最有效率的。从桌子 A 听一个菜,接着桌子 B 听一个菜,然后再桌子 A,然后再桌子 B 这样的流程会更加缓慢。出于同样原因,处理器在处理的数据彼此较近的时候(比如在栈上)比较远的时候(比如可能在堆上)能更好的工作。当你的代码调用一个函数时,传递给函数的值(包括可能指向堆上数据的指针)和函数的局部变量被压入栈中。当函数结束时,这些值被移出栈。跟踪哪部分代码正在使用堆上的哪些数据,最大限度的减少堆上的重复数据的数量,以及清理堆上不再使用的数据确保不会耗尽空间,这些问题正是所有权系统要处理的。一旦理解了所有权,你就不需要经常考虑栈和堆了,不过明白了所有权的主要目的就是为了管理堆数据,能够帮助解释为什么所有权要以这种方式工作。

2.1.所有权规则

首先,让我们看一下所有权的规则。当我们通过举例说明时,请谨记这些规则:

Rust 中的每一个值都有一个 所有者(owner)。值在任一时刻有且只有一个所有者。当所有者(变量)离开作用域,这个值将被丢弃。

2.2.变量作用域

既然我们已经掌握了基本语法,将不会在之后的例子中包含 ​​fn main() ​​ 代码,所以如果你是一路跟过来的,必须手动将之后例子的代码放入一个 ​​main​​ 函数中。这样,例子将显得更加简明,使我们可以关注实际细节而不是样板代码。

在所有权的第一个例子中,我们看看一些变量的 作用域scope)。作用域是一个项(item)在程序中有效的范围。假设有这样一个变量:

let s = "hello";

变量 ​​s​​ 绑定到了一个字符串字面值,这个字符串值是硬编码进程序代码中的。这个变量从声明的点开始直到当前 作用域 结束时都是有效的。示例 1 中的注释标明了变量 ​​s​​ 在何处是有效的。

                      // s 在这里无效, 它尚未声明
let s = "hello"; // 从此处起,s 是有效的

// 使用 s
// 此作用域已结束,s 不再有效

示例 1:一个变量和其有效的作用域

换句话说,这里有两个重要的时间点:

•当 s ​​进入作用域 ​​时,它就是有效的。

•这一直持续到它 离开作用域 为止

目前为止,变量是否有效与作用域的关系跟其他编程语言是类似的。现在我们在此基础上介绍 ​​String​​ 类型。

2.3.String 类型

为了演示所有权的规则,我们需要一个比基本数据类型都要复杂的数据类型。前面介绍的类型都是已知大小的,可以存储在栈中,并且当离开作用域时被移出栈,如果代码的另一部分需要在不同的作用域中使用相同的值,可以快速简单地复制它们来创建一个新的独立实例。不过我们需要寻找一个存储在堆上的数据来探索 Rust 是如何知道该在何时清理数据的。

我们会专注于 ​​String​​ 与所有权相关的部分。这些方面也同样适用于标准库提供的或你自己创建的其他复杂数据类型。

我们已经见过字符串字面值,即被硬编码进程序里的字符串值。字符串字面值是很方便的,不过它们并不适合使用文本的每一种场景。原因之一就是它们是不可变的。另一个原因是并非所有字符串的值都能在编写代码时就知道:例如,要是想获取用户输入并存储该怎么办呢?为此,Rust 有第二个字符串类型,​​String​​。这个类型管理被分配到堆上的数据,所以能够存储在编译时未知大小的文本。可以使用 ​​from​​ 函数基于字符串字面值来创建 ​​String​​,如下:

let s = String::from("hello");

这两个冒号 ​​::​​ 是运算符,允许将特定的 ​​from​​ 函数置于 ​​String​​ 类型的命名空间(namespace)下,而不需要使用类似 ​​string_from​​ 这样的名字。

可以 修改此类字符串 :

let mut s = String::from("hello");

s.push_str(", world!"); // push_str() 在字符串后追加字面值

println!("", s); // 将打印 `hello, world!`

那么这里有什么区别呢?为什么 ​​String​​ 可变而字面值却不行呢?区别在于两个类型对内存的处理上。

2.4.内存与分配

就字符串字面值来说,我们在编译时就知道其内容,所以文本被直接硬编码进最终的可执行文件中。这使得字符串字面值快速且高效。不过这些特性都只得益于字符串字面值的不可变性。不幸的是,我们不能为了每一个在编译时大小未知的文本而将一块内存放入二进制文件中,并且它的大小还可能随着程序运行而改变。

对于 ​​String​​ 类型,为了支持一个可变,可增长的文本片段,需要在堆上分配一块在编译时未知大小的内存来存放内容。这意味着:

•必须在运行时向内存分配器(memory allocator)请求内存。

•​​需要一个当我们处理完 String 时将内存返回给分配器的方法。​

第一部分由我们完成:当调用 ​​String::from​​ 时,它的实现 (implementation) 请求其所需的内存。这在编程语言中是非常通用的。

然而,第二部分实现起来就各有区别了。在有 垃圾回收garbage collectorGC)的语言中, GC 记录并清除不再使用的内存,而我们并不需要关心它。在大部分没有 GC 的语言中,识别出不再使用的内存并调用代码显式释放就是我们的责任了,跟请求内存的时候一样。从历史的角度上说正确处理内存回收曾经是一个困难的编程问题。如果忘记回收了会浪费内存。如果过早回收了,将会出现无效变量。如果重复回收,这也是个 bug。我们需要精确的为一个 ​​allocate​​ 配对一个 ​​free​​。

Rust 采取了一个不同的策略:内存在拥有它的变量离开作用域后就被自动释放。下面是示例 1 中作用域例子的一个使用 ​​String​​ 而不是字符串字面值的版本:


let s = String::from("hello"); // 从此处起,s 是有效的

// 使用 s
// 此作用域已结束,
// s 不再有效

这是一个将 ​​String​​ 需要的内存返回给分配器的很自然的位置:当 ​​s​​ 离开作用域的时候。当变量离开作用域,Rust 为我们调用一个特殊的函数。这个函数叫做 ​​drop​​,在这里 ​​String​​ 的作者可以放置释放内存的代码。Rust 在结尾的 ​​​​ 处自动调用 ​​drop​​。

注意:在 C++ 中,这种 item 在生命周期结束时释放资源的模式有时被称作 资源获取即初始化(Resource Acquisition Is Initialization (RAII))。如果你使用过 RAII 模式的话应该对 Rust 的 drop 函数并不陌生。

这个模式对编写 Rust 代码的方式有着深远的影响。现在它看起来很简单,不过在更复杂的场景下代码的行为可能是不可预测的,比如当有多个变量使用在堆上分配的内存时。现在让我们探索一些这样的场景。

2.4.1.变量与数据交互的方式(一):移动

在Rust 中,多个变量可以采取不同的方式与同一数据进行交互。让我们看看示例 2 中一个使用整型的例子。

let x = 5;
let y = x;

示例 2:将变量 x 的整数值赋给 y

我们大致可以猜到这在干什么:“将 ​​5​​ 绑定到 ​​x​​;接着生成一个值 ​​x​​ 的拷贝并绑定到 ​​y​​”。现在有了两个变量,​​x​​ 和 ​​y​​,都等于 ​​5​​。这也正是事实上发生了的,因为整数是有已知固定大小的简单值,所以这两个 ​​5​​ 被放入了栈中。

现在看看这个 ​​String​​ 版本:

let s1 = String::from("hello");
let s2 = s1;

这看起来与上面的代码非常类似,所以我们可能会假设他们的运行方式也是类似的:也就是说,第二行可能会生成一个 ​​s1​​ 的拷贝并绑定到 ​​s2​​ 上。不过,事实上并不完全是这样。

看看图1 以了解 ​​String​​ 的底层会发生什么。​​String​​ 由三部分组成,如图左侧所示:一个指向存放字符串内容内存的指针,一个长度,和一个容量。这一组数据存储在栈上。右侧则是堆上存放内容的内存部分。


【Rust学习】内存安全探秘:变量的所有权、引用与借用_Rust


图 1:将值 "hello" 绑定给 s1 的 String 在内存中的表现形式

长度表示 ​​String​​ 的内容当前使用了多少字节的内存。容量是 ​​String​​ 从分配器总共获取了多少字节的内存。长度与容量的区别是很重要的,不过在当前上下文中并不重要,所以现在可以忽略容量。

当我们将 ​​s1​​ 赋值给 ​​s2​​,​​String​​ 的数据被复制了,这意味着我们从栈上拷贝了它的指针、长度和容量。我们并没有复制指针指向的堆上数据。换句话说,内存中数据的表现如图2 所示。


【Rust学习】内存安全探秘:变量的所有权、引用与借用_Rust_02


图 2:变量 s2 的内存表现,它有一份 s1 指针、长度和容量的拷贝

这个表现形式看起来 并不像 图3 中的那样,如果 Rust 也拷贝了堆上的数据,那么内存看起来就是这样的。如果 Rust 这么做了,那么操作 ​​s2 = s1​​ 在堆上数据比较大的时候会对运行时性能造成非常大的影响。


【Rust学习】内存安全探秘:变量的所有权、引用与借用_Rust_03


图 3:另一个 s2 = s1 时可能的内存表现,如果 Rust 同时也拷贝了堆上的数据的话

之前我们提到过当变量离开作用域后,Rust 自动调用 ​​drop​​ 函数并清理变量的堆内存。不过图 2 展示了两个数据指针指向了同一位置。这就有了一个问题:当 ​​s2​​ 和 ​​s1​​ 离开作用域,他们都会尝试释放相同的内存。这是一个叫做 二次释放double free)的错误,也是之前提到过的内存安全性 bug 之一。两次释放(相同)内存会导致内存污染,它可能会导致潜在的安全漏洞。

为了确保内存安全,在 ​​let s2 = s1​​ 之后,Rust 认为 ​​s1​​ 不再有效,因此 Rust 不需要在 ​​s1​​ 离开作用域后清理任何东西。看看在 ​​s2​​ 被创建之后尝试使用 ​​s1​​ 会发生什么;这段代码不能运行:

let s1 = String::from("hello");
let s2 = s1;

println!(", world!", s1);

你会得到一个类似如下的错误,因为 Rust 禁止你使用无效的引用。

$ cargo run
Compiling ownership v0.1.0 (file:///projects/ownership)
error[E0382]: borrow of moved value: `s1`
--> src/main.rs:5:28
|
2 | let s1 = String::from("hello");
| -- move occurs because `s1` has type `String`, which does not implement the `Copy` trait
3 | let s2 = s1;
| -- value moved here
4 |
5 | println!(", world!", s1);
| ^^ value borrowed here after move

For more information about this error, try `rustc --explain E0382`.
error: could not compile `ownership` due to previous error

如果你在其他语言中听说过术语 浅拷贝shallow copy)和 深拷贝deep copy),那么拷贝指针、长度和容量而不拷贝数据可能听起来像浅拷贝。不过因为 Rust 同时使第一个变量无效了,这个操作被称为 移动move),而不是浅拷贝。上面的例子可以解读为 ​​s1​​被 移动 到了 ​​s2​​ 中。那么具体发生了什么,如图 4 所示。


【Rust学习】内存安全探秘:变量的所有权、引用与借用_作用域_04


图 4:s1 无效之后的内存表现

这样就解决了我们的问题!因为只有 ​​s2​​ 是有效的,当其离开作用域,它就释放自己的内存,完毕。

另外,这里还隐含了一个设计选择:Rust 永远也不会自动创建数据的 “深拷贝”。因此,任何 自动 的复制可以被认为对运行时性能影响较小。

2.4.2.变量与数据交互的方式(二):克隆

如果我们 确实 需要深度复制 ​​String​​ 中堆上的数据,而不仅仅是栈上的数据,可以使用一个叫做 ​​clone​​ 的通用函数。第五章会讨论方法语法,不过因为方法在很多语言中是一个常见功能,所以之前你可能已经见过了。

这是一个实际使用 ​​clone​​ 方法的例子:

let s1 = String::from("hello");
let s2 = s1.clone();

println!("s1 = , s2 = ", s1, s2);


这段代码能正常运行,并且明确产生图 3 中行为,这里堆上的数据 确实 被复制了。

当出现 ​​clone​​ 调用时,你知道一些特定的代码被执行而且这些代码可能相当消耗资源。你很容易察觉到一些不寻常的事情正在发生。

2.4.3.只在栈上的数据:拷贝

这里还有一个没有提到的小窍门。这些代码使用了整型并且是有效的,他们是示例 2 中的一部分:

let x = 5;
let y = x;

println!("x = , y = ", x, y);

但这段代码似乎与我们刚刚学到的内容相矛盾:没有调用 ​​clone​​,不过 ​​x​​ 依然有效且没有被移动到 ​​y​​ 中。

原因是像整型这样的在编译时已知大小的类型被整个存储在栈上,所以拷贝其实际的值是快速的。这意味着没有理由在创建变量 ​​y​​ 后使 ​​x​​ 无效。换句话说,这里没有深浅拷贝的区别,所以这里调用 ​​clone​​ 并不会与通常的浅拷贝有什么不同,我们可以不用管它。

Rust 有一个叫做 ​​Copy​​ trait 的特殊注解,可以用在类似整型这样的存储在栈上的类型上。如果一个类型实现了 ​​Copy​​ trait,那么一个旧的变量在将其赋值给其他变量后仍然可用。

Rust 不允许自身或其任何部分实现了 ​​Drop​​ trait 的类型使用 ​​Copy​​ trait。如果我们对其值离开作用域时需要特殊处理的类型使用 ​​Copy​​ 注解,将会出现一个编译时错误。

那么哪些类型实现了 ​​Copy​​ trait 呢?你可以查看给定类型的文档来确认,不过作为一个通用的规则,任何一组简单标量值的组合都可以实现 ​​Copy​​,任何不需要分配内存或某种形式资源的类型都可以实现 ​​Copy​​ 。如下是一些 ​​Copy​​ 的类型:

•所有整数类型,比如 u32。

•布尔类型,bool,它的值是 true 和 false。

•所有浮点数类型,比如 f64。

•字符类型,char。

•元组,当且仅当其包含的类型也都实现 Copy 的时候。比如,(i32, i32) 实现了 Copy,但 (i32, String) 就没有。

2.5.所有权与函数

将值传递给函数与给变量赋值的原理相似。向函数传递值可能会移动或者复制,就像赋值语句一样。示例 3 使用注释展示变量何时进入和离开作用域:

文件名: src/main.rs

fn main() 
let s = String::from("hello"); // s 进入作用域

takes_ownership(s); // s 的值移动到函数里 ...
// ... 所以到这里不再有效

let x = 5; // x 进入作用域

makes_copy(x); // x 应该移动函数里,
// 但 i32 是 Copy 的,
// 所以在后面可继续使用 x

// 这里, x 先移出了作用域,然后是 s。但因为 s 的值已被移走,
// 没有特殊之处

fn takes_ownership(some_string: String) // some_string 进入作用域
println!("", some_string);
// 这里,some_string 移出作用域并调用 `drop` 方法。
// 占用的内存被释放

fn makes_copy(some_integer: i32) // some_integer 进入作用域
println!("", some_integer);
// 这里,some_integer 移出作用域。没有特殊之处

示例 3:带有所有权和作用域注释的函数

当尝试在调用 ​​takes_ownership​​ 后使用 ​​s​​ 时,Rust 会抛出一个编译时错误。这些静态检查使我们免于犯错。试试在 ​​main​​ 函数中添加使用 ​​s​​ 和 ​​x​​ 的代码来看看哪里能使用他们,以及所有权规则会在哪里阻止我们这么做。

2.6.返回值与作用域

返回值也可以转移所有权。示例 4 展示了一个返回了某些值的示例,与示例 3 一样带有类似的注释。

文件名: src/main.rs

fn main() 
let s1 = gives_ownership(); // gives_ownership 将返回值
// 转移给 s1

let s2 = String::from("hello"); // s2 进入作用域

let s3 = takes_and_gives_back(s2); // s2 被移动到
// takes_and_gives_back 中,
// 它也将返回值移给 s3
// 这里, s3 移出作用域并被丢弃。s2 也移出作用域,但已被移走,
// 所以什么也不会发生。s1 离开作用域并被丢弃

fn gives_ownership() -> String // gives_ownership 会将
// 返回值移动给
// 调用它的函数

let some_string = String::from("yours"); // some_string 进入作用域.

some_string // 返回 some_string
// 并移出给调用的函数
//


// takes_and_gives_back 将传入字符串并返回该值
fn takes_and_gives_back(a_string: String) -> String // a_string 进入作用域
//

a_string // 返回 a_string 并移出给调用的函数

示例 4: 转移返回值的所有权

变量的所有权总是遵循相同的模式:将值赋给另一个变量时移动它。当持有堆中数据值的变量离开作用域时,其值将通过 ​​drop​​ 被清理掉,除非数据被移动为另一个变量所有。

虽然这样是可以的,但是在每一个函数中都获取所有权并接着返回所有权有些啰嗦。如果我们想要函数使用一个值但不获取所有权该怎么办呢?如果我们还要接着使用它的话,每次都传进去再返回来就有点烦人了,除此之外,我们也可能想返回函数体中产生的一些数据。

我们可以使用元组来返回多个值,如示例 5 所示。

文件名: src/main.rs

fn main() 
let s1 = String::from("hello");

let (s2, len) = calculate_length(s1);

println!("The length of is .", s2, len);


fn calculate_length(s: String) -> (String, usize)
let length = s.len(); // len() 返回字符串的长度

(s, length)

示例 5: 返回参数的所有权

但是这未免有些形式主义,而且这种场景应该很常见。幸运的是,Rust 对此提供了一个不用获取所有权就可以使用值的功能,叫做 引用references)。

三.引用与借用

示例 5 中的元组代码有这样一个问题:我们必须将 ​​String​​ 返回给调用函数,以便在调用 ​​calculate_length​​ 后仍能使用 ​​String​​,因为 ​​String​​ 被移动到了 ​​calculate_length​​ 内。相反我们可以提供一个 ​​String​​ 值的引用(reference)。引用reference)像一个指针,因为它是一个地址,我们可以由此访问储存于该地址的属于其他变量的数据。 与指针不同,引用确保指向某个特定类型的有效值。

下面是如何定义并使用一个(新的)​​calculate_length​​ 函数,它以一个对象的引用作为参数而不是获取值的所有权:

文件名: src/main.rs

fn main() 
let s1 = String::from("hello");

let len = calculate_length(&s1);

println!("The length of is .", s1, len);


fn calculate_length(s: &String) -> usize
s.len()

首先,注意变量声明和函数返回值中的所有元组代码都消失了。其次,注意我们传递 ​​&s1​​ 给 ​​calculate_length​​,同时在函数定义中,我们获取 ​​&String​​ 而不是 ​​String​​。这些 & 符号就是 引用,它们允许你使用值但不获取其所有权。图 5 展示了一张示意图。


【Rust学习】内存安全探秘:变量的所有权、引用与借用_string_05


图 5:&String s 指向 String s1 示意图

注意:与使用 & 引用相反的操作是 解引用(dereferencing),它使用解引用运算符,*。我们将会在第八章遇到一些解引用运算符,并在第十五章详细讨论解引用。

仔细看看这个函数调用:

let s1 = String::from("hello");

let len = calculate_length(&s1);

​&s1​​ 语法让我们创建一个 指向 值 ​​s1​​ 的引用,但是并不拥有它。因为并不拥有这个值,所以当引用停止使用时,它所指向的值也不会被丢弃。

同理,函数签名使用 ​​&​​ 来表明参数 ​​s​​ 的类型是一个引用。让我们增加一些解释性的注释:

fn calculate_length(s: &String) -> usize  // s是String的引用
s.len()
// 这里,s 离开了作用域。但因为它并不拥有引用值的所有权,
// 所以什么也不会发生

变量 ​​s​​ 有效的作用域与函数参数的作用域一样,不过当 ​​s​​ 停止使用时并不丢弃引用指向的数据,因为 ​​s​​ 并没有所有权。当函数使用引用而不是实际值作为参数,无需返回值来交还所有权,因为就不曾拥有所有权。

我们将创建一个引用的行为称为 借用borrowing)。正如现实生活中,如果一个人拥有某样东西,你可以从他那里借来。当你使用完毕,必须还回去。我们并不拥有它。

如果我们尝试修改借用的变量呢?尝试示例 6 中的代码。剧透:这行不通!

文件名: src/main.rs

fn main() 
let s = String::from("hello");

change(&s);


fn change(some_string: &String)
some_string.push_str(", world");

示例 6:尝试修改借用的值

这里是错误:

$ cargo run
Compiling ownership v0.1.0 (file:///projects/ownership)
error[E0596]: cannot borrow `*some_string` as mutable, as it is behind a `&` reference
--> src/main.rs:8:5
|
7 | fn change(some_string: &String)
| ------- help: consider changing this to be a mutable reference: `&mut String`
8 | some_string.push_str(", world");
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `some_string` is a `&` reference, so the data it refers to cannot be borrowed as mutable

For more information about this error, try `rustc --explain E0596`.
error: could not compile `ownership` due to previous error

正如变量默认是不可变的,引用也一样。(默认)不允许修改引用的值。

3.1.可变引用

我们通过一个小调整就能修复示例 6 代码中的错误,允许我们修改一个借用的值,这就是 可变引用mutable reference):

文件名: src/main.rs

fn main() 
let mut s = String::from("hello");

change(&mut s);


fn change(some_string: &mut String)
some_string.push_str(", world");

首先,我们必须将 ​​s​​ 改为 ​​mut​​。然后在调用 ​​change​​ 函数的地方创建一个可变引用 ​​&mut s​​,并更新函数签名以接受一个可变引用 ​​some_string: &mut String​​。这就非常清楚地表明,​​change​​ 函数将改变它所借用的值。

可变引用有一个很大的限制:如果你有一个对该变量的可变引用,你就不能再创建对该变量的引用。这些尝试创建两个 ​​s​​ 的可变引用的代码会失败:

文件名: src/main.rs

let mut s = String::from("hello");

let r1 = &mut s;
let r2 = &mut s;

println!(", ", r1, r2);

错误如下:

$ cargo run
Compiling ownership v0.1.0 (file:///projects/ownership)
error[E0499]: cannot borrow `s` as mutable more than once at a time
--> src/main.rs:5:14
|
4 | let r1 = &mut s;
| ------ first mutable borrow occurs here
5 | let r2 = &mut s;
| ^^^^^^ second mutable borrow occurs here
6 |
7 | println!(", ", r1, r2);
| -- first borrow later used here

For more information about this error, try `rustc --explain E0499`.
error: could not compile `ownership` due to previous error

这个报错说这段代码是无效的,因为我们不能在同一时间多次将 ​​s​​ 作为可变变量借用。第一个可变的借入在 ​​r1​​ 中,并且必须持续到在 ​​println!​​ 中使用它,但是在那个可变引用的创建和它的使用之间,我们又尝试在 ​​r2​​ 中创建另一个可变引用,该引用借用与 ​​r1​​ 相同的数据。

这一限制以一种非常小心谨慎的方式允许可变性,防止同一时间对同一数据存在多个可变引用。新 Rustacean 们经常难以适应这一点,因为大部分语言中变量任何时候都是可变的。这个限制的好处是 Rust 可以在编译时就避免数据竞争。数据竞争data race)类似于竞态条件,它可由这三个行为造成:

•两个或更多指针同时访问同一数据。

•至少有一个指针被用来写入数据。

•没有同步数据访问的机制。

数据竞争会导致未定义行为,难以在运行时追踪,并且难以诊断和修复;Rust 避免了这种情况的发生,因为它甚至不会编译存在数据竞争的代码!

一如既往,可以使用大括号来创建一个新的作用域,以允许拥有多个可变引用,只是不能 同时 拥有:

let mut s = String::from("hello");


let r1 = &mut s;
// r1 在这里离开了作用域,所以我们完全可以创建一个新的引用

let r2 = &mut s;

Rust 在同时使用可变与不可变引用时也采用的类似的规则。这些代码会导致一个错误:

let mut s = String::from("hello");

let r1 = &s; // 没问题
let r2 = &s; // 没问题
let r3 = &mut s; // 大问题

println!(", , and ", r1, r2, r3);

错误如下:

$ cargo run
Compiling ownership v0.1.0 (file:///projects/ownership)
error[E0502]: cannot borrow `s` as mutable because it is also borrowed as immutable
--> src/main.rs:6:14
|
4 | let r1 = &s; // no problem
| -- immutable borrow occurs here
5 | let r2 = &s; // no problem
6 | let r3 = &mut s; // BIG PROBLEM
| ^^^^^^ mutable borrow occurs here
7 |
8 | println!(", , and ", r1, r2, r3);
| -- immutable borrow later used here

For more information about this error, try `rustc --explain E0502`.
error: could not compile `ownership` due to previous error

错误提示我们也不能在拥有不可变引用的同时拥有可变引用。

不可变引用的用户可不希望在他们的眼皮底下值就被意外的改变了!然而,多个不可变引用是可以的,因为没有哪个只能读取数据的人有能力影响其他人读取到的数据。

注意一个引用的作用域从声明的地方开始一直持续到最后一次使用为止。例如,因为最后一次使用不可变引用(​​println!​​),发生在声明可变引用之前,所以如下代码是可以编译的:

let mut s = String::from("hello");

let r1 = &s; // 没问题
let r2 = &s; // 没问题
println!(" and ", r1, r2);
// 此位置之后 r1 和 r2 不再使用

let r3 = &mut s; // 没问题
println!("", r3);

不可变引用 ​​r1​​ 和 ​​r2​​ 的作用域在 ​​println!​​ 最后一次使用之后结束,这也是创建可变引用 ​​r3​​ 的地方。它们的作用域没有重叠,所以代码是可以编译的。编译器在作用域结束之前判断不再使用的引用的能力被称为 非词法作用域生命周期Non-Lexical Lifetimes,简称 NLL)。

尽管这些错误有时使人沮丧,但请牢记这是 Rust 编译器在提前指出一个潜在的 bug(在编译时而不是在运行时)并精准显示问题所在。这样你就不必去跟踪为何数据并不是你想象中的那样。

3.2.悬垂引用(Dangling References)

在具有指针的语言中,很容易通过释放内存时保留指向它的指针而错误地生成一个 悬垂指针dangling pointer),所谓悬垂指针是其指向的内存可能已经被分配给其它持有者。相比之下,在 Rust 中编译器确保引用永远也不会变成悬垂状态:当你拥有一些数据的引用,编译器确保数据不会在其引用之前离开作用域。

让我们尝试创建一个悬垂引用,Rust 会通过一个编译时错误来避免:

文件名: src/main.rs

fn main() 
let reference_to_nothing = dangle();


fn dangle() -> &String
let s = String::from("hello");

&s

这里是错误:

$ cargo run
Compiling ownership v0.1.0 (file:///projects/ownership)
error[E0106]: missing lifetime specifier
--> src/main.rs:5:16
|
5 | fn dangle() -> &String
| ^ expected named lifetime parameter
|
= help: this functions return type contains a borrowed value, but there is no value for it to be borrowed from
help: consider using the `static` lifetime
|
5 | fn dangle() -> &static String
| ~~~~~~~~

For more information about this error, try `rustc --explain E0106`.
error: could not compile `ownership` due to previous error

错误信息引用了一个我们还未介绍的功能:生命周期(lifetimes)。第十章会详细介绍生命周期。不过,如果你不理会生命周期部分,错误信息中确实包含了为什么这段代码有问题的关键信息:

this functions return type contains a borrowed value, but there is no value
for it to be borrowed from

让我们仔细看看我们的 ​​dangle​​ 代码的每一步到底发生了什么:

文件名: src/main.rs

fn dangle() -> &String  // dangle 返回一个字符串的引用

let s = String::from("hello"); // s 是一个新字符串

&s // 返回字符串 s 的引用
// 这里 s 离开作用域并被丢弃。其内存被释放。
// 危险!

因为 ​​s​​ 是在 ​​dangle​​ 函数内创建的,当 ​​dangle​​ 的代码执行完毕后,​​s​​ 将被释放。不过我们尝试返回它的引用。这意味着这个引用会指向一个无效的 ​查看详情

rust编程语言入门之高级特性(代码片段)

...不安全Rust匹配命名变量隐藏着第二个语言,它没有强制内存安全保证:UnsafeRust(不安全的Rust)和普通的Rust一样,但提供了额外的“超能力”UnsafeRust存在的原因:静态分析是保守的。使用UnsafeRust:我知道自己在做什么,并承担... 查看详情

rust内存安全--借用(代码片段)

rust的内存安全在编译期间就可以被检查出来。关于借用,总结一句话是“共享不可变,可变不共享”譬如这段代码,是可以通过编译的fnmain()leti=1;letp1=&i;letp2=&i;println!("",i,p1,p2);虽然发送了共享... 查看详情

内存安全

...没有垃圾回收机制,却成功实现了内存安全(memorysafety)。所有权在Rust中,所有权(ownership)系统是零成本抽象(zero-costabstraction)的一个主要例子。对所有权的分析是在编译阶段就完成的,并不带来任何运行时成本(run-timecost)。默认情... 查看详情

rust内存安全--借用(代码片段)

rust的内存安全在编译期间就可以被检查出来。关于借用,总结一句话是“共享不可变,可变不共享”譬如这段代码,是可以通过编译的fnmain()leti=1;letp1=&i;letp2=&i;println!("",i,p1,p2);虽然发送了共享... 查看详情

rust内存安全--借用(代码片段)

rust的内存安全在编译期间就可以被检查出来。关于借用,总结一句话是“共享不可变,可变不共享”譬如这段代码,是可以通过编译的fnmain()leti=1;letp1=&i;letp2=&i;println!("",i,p1,p2);虽然发送了共享... 查看详情

rust编程语言入门之智能指针(代码片段)

...能指针Box<T>:在heap内存上分配值Rc<T>:启用多重所有权的引用计数类型Ref<T>和RefMut<T>,通过RefCell<T>访问:在运行时而不是编译时强制借用规则的类型此外:内部可变模型(interiormutab 查看详情

rust学习笔记1.基础语法(代码片段)

...体的语句和表达式函数返回值7、条件语句8、循环语句9、所有权Rust的所有权内存的分配变量与数据交互方式移动克隆垂悬引用10、切片11、结构体12、枚举类match语法Option枚举类iflet语法13、生命周期生命周期注释函数的生命周期... 查看详情

我的第一篇rust博客(代码片段)

...rust的内存安全和线程安全实现机制就是独占内存资源的所有权,任何时候只允许只有一个变量(可以是变量的引用)对内存进行修改,并且引用是有生命周期的,编译器会约束引用的使用,它在原始变量释放内存之前必须归还... 查看详情

java内存模型探秘

1.Java内存模型概述  Java内存模型是一种抽象概念,不是真实存在的。主要定义了程序中各个变量的访问规则,即在虚拟机中将变量存储到内存和从内存取出变量这样的底层细节。注意:这里的变量仅包括实例字段、静态字段... 查看详情

usingrust-gdbtodebugrust(byquqi99)(代码片段)

...内存安全,Rust建立了严格的安全内存管理模型:所有权系统,每个被分配的内存都有一个独占其所有权的指针。借用和生命周期,每个变量都有其 查看详情

[易学易懂系列|rustlang语言|零基础|快速入门|(21)|智能指针](代码片段)

...址外,还有额外的其他属性或元数据。在Rust中,因为有所有权和借用的概念,所以引用和智能指针,又有一点不一样。简单来说,智能指针,拥有数据所有权,而引用没有。智能指针分以下几种:1.Box,用于在堆里分配内存。2.Rc,... 查看详情

内存线程安全与并发

@内存机制引用自一、java内存机制java程序在内存中的分配有4种,分别是:全局数据区:保存static修饰的属性;全局代码区:保存static修饰的静态方法;栈内存空间:保存所有的对象名称,这些对象名称指向对象所在的堆内存空... 查看详情

rust为什么我建议你学一下rust|rust初探(代码片段)

...3;+,但可以通过使用借用检查器来验证引用来保证内存安全。Rust在没有垃圾收集的情况下实现了内存安全,并且引用计数是 查看详情

新手眼中的rust所有权规则(代码片段)

新手眼中的Rust所有权规则如果你有关注本人博客,那么很明显,从今年年初开始,我便开始学习Rust。此文与之前风格略有不同,旨在总结阅读Rust书籍时遇到的要点。到目前为止,它包含了我对Rust所有权规则的所有理解。Rust的... 查看详情

rust所有权语义模型

首发于知乎专栏本文试图从语义角度来解释Rust所有权的概念,以便帮助降低Rust的学习曲线。编程语言的内存管理,大概可以分为自动和手动两种。自动管理就是用GC(垃圾回收)来自动管理内存,像Java、Ruby、Golang、Elixir等语言... 查看详情

怎样才能强制 Rust 获得分配的内存的所有权,而不是通过其安全方法分配的内存?

】怎样才能强制Rust获得分配的内存的所有权,而不是通过其安全方法分配的内存?【英文标题】:HowcanoneforceRusttotakeownershipofmemoryallocatedotherthanbyitssafemethods?【发布时间】:2019-07-1721:30:41【问题描述】:WillCrichton在2018年2月题为... 查看详情

#yyds干货盘点#为什么要学习rust?

1.Rust相关概念与简介Rust是一种令人兴奋的新兴编程语言,它可以让每个人编写可靠且高效的软件。Rust可以用来替换C/C++,Rust和他们具有相同的性能,但是很多常见的bug在编译时就可以被消灭。Rust是一种通用的编程语言,但是它... 查看详情