mirror of
https://github.com/KaiserY/trpl-zh-cn
synced 2024-11-14 21:11:31 +08:00
Merge pull request #249 from xiaohulu/master
update ch04-01-what-is-ownership.md
This commit is contained in:
commit
0e052b2116
@ -2,143 +2,139 @@
|
|||||||
|
|
||||||
> [ch04-01-what-is-ownership.md](https://github.com/rust-lang/book/blob/master/second-edition/src/ch04-01-what-is-ownership.md)
|
> [ch04-01-what-is-ownership.md](https://github.com/rust-lang/book/blob/master/second-edition/src/ch04-01-what-is-ownership.md)
|
||||||
> <br>
|
> <br>
|
||||||
> commit ec65990849230388e4ce4db5b7a0cb8a0f0d60e2
|
> commit f949ff883628db8ed2f2f5f19e146ebf19ed6a6f
|
||||||
|
|
||||||
Rust 的核心功能(之一)是 **所有权**(*ownership*)。虽然该功能很容易解释,但它对语言的其他部分有深刻的影响。
|
Rust 的核心功能(之一)是 **所有权**(*ownership*)。虽然该功能很容易解释,但它对语言的其他部分有着深刻的影响。
|
||||||
|
|
||||||
所有程序都必须管理其运行时使用计算机内存的方式。一些语言中使用垃圾回收在程序运行过程中来时刻寻找不再被使用的内存;在另一些语言中,程序员必须亲自分配和释放内存。Rust 则选择了第三种方式:内存被一个所有权系统管理,它拥有一系列的规则使编译器在编译时进行检查。任何所有权系统的功能都不会导致运行时开销。
|
所有运行的程序都必须管理其使用计算机内存的方式。一些语言中具有垃圾回收机制,在程序运行时不断地寻找不再使用的内存;在另一些语言中,程序员必须亲自分配和释放内存。Rust 则选择了第三种方式:通过所有权系统管理内存,编译器在编译时会根据一系列的规则进行检查。在运行时,所有权系统的任何功能都不会减慢程序。
|
||||||
|
|
||||||
因为所有权对很多程序员来说都是一个新概念,需要一些时间来适应。好消息是随着你对 Rust 和所有权系统的规则越来越有经验,你就越能自然地编写出安全和高效的代码。持之以恒!
|
因为所有权对很多程序员来说都是一个新概念,需要一些时间来适应。好消息是随着你对 Rust 和所有权系统的规则越来越有经验,你就越能自然地编写出安全和高效的代码。持之以恒!
|
||||||
|
|
||||||
当你理解了所有权,你将有一个坚实的基础来理解使Rust独特的功能。在本章中,你将通过一些常见数据结构的例子来学习所有权:字符串。
|
当你理解了所有权,你将有一个坚实的基础来理解那些使 Rust 独特的功能。在本章中,你将通过完成一些示例来学习所有权,这些示例基于一个常用的数据结构:字符串。
|
||||||
|
|
||||||
<!-- PROD: START BOX -->
|
|
||||||
|
|
||||||
> ### 栈(Stack)与堆(Heap)
|
> ### 栈(Stack)与堆(Heap)
|
||||||
>
|
>
|
||||||
> 在很多语言中并不经常需要考虑到栈与堆。不过在像 Rust 这样的系统编程语言中,值是位于栈上还是堆上在更大程度上影响了语言的行为以及为何必须做出这样的选择。我们会在本章的稍后部分描述所有权与堆与栈相关的部分,所以这里只是一个用来预热的简要解释。
|
> 在很多语言中,你并不需要经常考虑到栈与堆。不过在像 Rust 这样的系统编程语言中,值是位于栈上还是堆上在更大程度上影响了语言的行为以及为何必须做出这样的抉择。我们会在本章的稍后部分描述所有权与栈和堆相关的内容,所以这里只是一个用来预热的简要解释。
|
||||||
>
|
>
|
||||||
> 栈和堆都是代码在运行时可供使用的内存部分,不过他们以不同的结构组成。栈以放入值的顺序存储并以相反顺序取出值。这也被称作 **后进先出**(*last in, first out*)。想象一下一叠盘子:当增加更多盘子时,把他们放在盘子堆的顶部,当需要盘子时,也从顶部拿走。不能从中间也不能从底部增加或拿走盘子!增加数据叫做 **进栈**(*pushing onto the stack*),而移出数据叫做 **出栈**(*popping off the stack*)。
|
> 栈和堆都是代码在运行时可供使用的内存,但是它们的结构不同。栈以放入值的顺序存储值并以相反顺序取出值。这也被称作 **后进先出**(*last in, first out*)。想象一下一叠盘子:当增加更多盘子时,把它们放在盘子堆的顶部,当需要盘子时,也从顶部拿走。不能从中间也不能从底部增加或拿走盘子!增加数据叫做 **进栈**(*pushing onto the stack*),而移出数据叫做 **出栈**(*popping off the stack*)。
|
||||||
>
|
>
|
||||||
> 栈的操作是十分快速的,这主要是得益于它存取数据的方式:因为数据存取的位置总是在栈顶而不需要寻找一个位置存放或读取数据。另一个栈的属性也让操作栈中数据快速,是因为栈中的所有数据都必须有一个已知且固定的大小。
|
> 栈的操作是十分快速的,这主要是得益于它存取数据的方式:因为数据存取的位置总是在栈顶而不需要寻找一个位置存放或读取数据。另一个让操作栈快速的属性是,栈中的所有数据都必须占用已知且固定的大小。
|
||||||
>
|
>
|
||||||
> 相反对于在编译时未知大小或大小可能变化的数据,可以把他们储存在堆上。堆是缺乏组织的:当向堆放入数据时,我们请求一定大小的空间。操作系统在堆的某处找到一块足够大的空位,把它标记为已使用,并返回给我们一个其位置的 **指针**(*pointer*)。这个过程称作 **在堆上分配内存**(*allocating on the heap*),并且有时这个过程就简称为 “分配”(allocating)。向栈中放入数据并不被认为是分配。因为指针是已知的固定大小的,我们可以将指针储存在栈上,不过当需要实际数据时,必须访问指针。
|
> 在编译时大小未知或大小可能变化的数据,要改为存储在堆上。堆是缺乏组织的:当向堆放入数据时,你要请求一定大小的空间。操作系统在堆的某处找到一块足够大的空位,把它标记为已使用,并返回一个表示该位置地址的 **指针**(*pointer*)。这个过程称作 **在堆上分配内存**(*allocating on the heap*),有时简称为 “分配”(allocating)。将数据推入栈中并不被认为是分配。因为指针的大小是已知并且固定的,你可以将指针存储在栈上,不过当需要实际数据时,必须访问指针。
|
||||||
>
|
>
|
||||||
> 想象一下去餐馆就坐吃饭。当进入时,你说明有几个人,餐馆员工会找到一个够大的空桌子并领你们过去。如果有人来迟了,他们也可以通过询问来找到你们坐在哪。
|
> 想象一下去餐馆就座吃饭。当进入时,你说明有几个人,餐馆员工会找到一个够大的空桌子并领你们过去。如果有人来迟了,他们也可以通过询问来找到你们坐在哪。
|
||||||
>
|
>
|
||||||
> 访问堆上的数据要比访问栈上的数据要慢因为必须通过指针来访问。现代处理器在内存中跳转越少就越快(缓存)。继续类比,假设有一个服务员在餐厅里处理多个桌子的点菜。在一个桌子报完所有菜后再移动到下一个桌子是最有效率的。从桌子 A 听一个菜,接着桌子 B 听一个菜,然后再桌子 A,然后再桌子 B 这样的流程会更加缓慢。出于同样原因,处理器在处理的数据之间彼此较近的时候(比如在栈上)比较远的时候(比如可能在堆上)能更好的工作。在堆上分配大量的空间也可能消耗时间。
|
> 访问堆上的数据比访问栈上的数据慢,因为必须通过指针来访问。现代处理器在内存中跳转越少就越快(缓存)。继续类比,假设有一个服务员在餐厅里处理多个桌子的点菜。在一个桌子报完所有菜后再移动到下一个桌子是最有效率的。从桌子 A 听一个菜,接着桌子 B 听一个菜,然后再桌子 A,然后再桌子 B 这样的流程会更加缓慢。出于同样原因,处理器在处理的数据彼此较近的时候(比如在栈上)比较远的时候(比如可能在堆上)能更好的工作。在堆上分配大量的空间也可能消耗时间。
|
||||||
>
|
>
|
||||||
> 当调用一个函数,传递给函数的值(包括可能指向堆上数据的指针)和函数的局部变量被压入栈中。当函数结束时,这些值被移出栈。
|
> 当你的代码调用一个函数时,传递给函数的值(包括可能指向堆上数据的指针)和函数的局部变量被压入栈中。当函数结束时,这些值被移出栈。
|
||||||
>
|
>
|
||||||
> 记录何处的代码在使用堆上的什么数据,最小化堆上的冗余数据的数量以及清理堆上不再使用的数据以致不至于耗尽空间,这些所有的问题正是所有权系统要处理的。一旦理解了所有权,你就不需要经常考虑栈和堆了,不过理解如何管理堆内存可以帮助我们理解所有权为何存在以及为什么要以这种方式工作。
|
> 跟踪哪部分代码正在使用堆上的哪些数据,最大限度的减少堆上的重复数据的数量,以及清理堆上不再使用的数据确保不会耗尽空间,这些问题正是所有权系统要处理的。一旦理解了所有权,你就不需要经常考虑栈和堆了,不过明白了所有权的存在就是为了管理堆数据,能够帮助解释为什么所有权要以这种方式工作。
|
||||||
|
|
||||||
<!-- PROD: END BOX -->
|
|
||||||
|
|
||||||
### 所有权规则
|
### 所有权规则
|
||||||
|
|
||||||
首先,让我们看一下所有权的规则。请记住它们,我们将讲解一些它们的例子:
|
首先,让我们看一下所有权的规则。当我们通过举例说明时,请谨记这些规则:
|
||||||
|
|
||||||
> 1. Rust 中每一个值都有一个称之为其 **所有者**(*owner*)的变量。
|
> 1. Rust 中的每一个值都有一个被称为其 **所有者**(*owner*)的变量。
|
||||||
> 2. 值有且只能有一个所有者。
|
> 2. 值有且只有一个所有者。
|
||||||
> 3. 当所有者(变量)离开作用域,这个值将被丢弃。
|
> 3. 当所有者(变量)离开作用域,这个值将被丢弃。
|
||||||
|
|
||||||
### 变量作用域
|
### 变量作用域
|
||||||
|
|
||||||
我们已经在第二章完成过一个 Rust 程序的例子。现在我们已经掌握了基本语法,所以不会在之后的例子中包含 `fn main() {` 代码了,所以如果你是一路跟过来的,必须手动将之后例子的代码放入一个 `main` 函数中。为此,例子将显得更加简明,使我们可以关注具体细节而不是样板代码。
|
我们已经在第二章完成一个 Rust 程序示例。既然我们已经掌握了基本语法,将不会在之后的例子中包含 `fn main() {` 代码,所以如果你是一路跟过来的,必须手动将之后例子的代码放入一个 `main` 函数中。这样,例子将显得更加简明,使我们可以关注实际细节而不是样板代码。
|
||||||
|
|
||||||
作为所有权的第一个例子,我们看看一些变量的 **作用域**(*scope*)。作用域是一个项(原文:item) 在程序中有效的范围。假设有这样一个变量:
|
在所有权的第一个例子中,我们看看一些变量的 **作用域**(*scope*)。作用域是一个项(item)在程序中有效的范围。假设有这样一个变量:
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
let s = "hello";
|
let s = "hello";
|
||||||
```
|
```
|
||||||
|
|
||||||
变量 `s` 绑定到了一个字符串字面值,这个字符串值是硬编码进程序代码中的。这个变量从声明的点开始直到当前 **作用域** 结束时都是有效的。示例 4-1 的注释标明了变量 `s` 在何处是有效的:
|
变量 `s` 绑定到了一个字符串字面值,这个字符串值是硬编码进程序代码中的。这个变量从声明的点开始直到当前 **作用域** 结束时都是有效的。示例 4-1 的注释标明了变量 `s` 在何处是有效的。
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
{ // s is not valid here, it’s not yet declared
|
{ // s 在这里无效, 它尚未声明
|
||||||
let s = "hello"; // s is valid from this point forward
|
let s = "hello"; // 从此处起,s 是有效的
|
||||||
|
|
||||||
// do stuff with s
|
// 使用 s
|
||||||
} // this scope is now over, and s is no longer valid
|
} // 此作用域已结束,s 不再有效
|
||||||
```
|
```
|
||||||
|
|
||||||
<span class="caption">示例 4-1:一个变量和其有效的作用域</span>
|
<span class="caption">示例 4-1:一个变量和其有效的作用域</span>
|
||||||
|
|
||||||
换句话说,这里有两个重要的点:
|
换句话说,这里有两个重要的时间点:
|
||||||
|
|
||||||
1. 当 `s` **进入作用域** 时,它就是有效的。
|
* 当 `s` **进入作用域** 时,它就是有效的。
|
||||||
2. 这一直持续到它 **离开作用域** 为止。
|
* 这一直持续到它 **离开作用域** 为止。
|
||||||
|
|
||||||
目前为止,变量是否有效与作用域的关系跟其他编程语言是类似的。现在我们在此基础上介绍 `String` 类型。
|
目前为止,变量是否有效与作用域的关系跟其他编程语言是类似的。现在我们在此基础上介绍 `String` 类型。
|
||||||
|
|
||||||
### `String` 类型
|
### `String` 类型
|
||||||
|
|
||||||
为了演示所有权的规则,我们需要一个比第三章讲到的任何一个都要复杂的数据类型。“Data Types” 部分涉及到的数据类型都是储存在栈上的并且当离开作用域时被移出栈,不过我们需要寻找一个储存在堆上的数据来探索 Rust 是如何知道该在何时清理数据的。
|
为了演示所有权的规则,我们需要一个比第三章 “数据类型” 中讲到的都要复杂的数据类型。前面介绍的类型都是存储在栈上的并且当离开作用域时被移出栈,不过我们需要寻找一个存储在堆上的数据来探索 Rust 是如何知道该在何时清理数据的。
|
||||||
|
|
||||||
这里使用 `String` 作为例子并专注于 `String` 与所有权相关的部分。这些方面也同样适用于其他标准库提供的或你自己创建的复杂数据类型。在第八章会更深入地讲解 `String`。
|
这里使用 `String` 作为例子,并专注于 `String` 与所有权相关的部分。这些方面也同样适用于标准库提供的或你自己创建的其他复杂数据类型。在第八章会更深入地讲解 `String`。
|
||||||
|
|
||||||
我们已经见过字符串字面值了,它被硬编码进程序里。字符串字面值是很方便的,不过他们并不总是适合所有需要使用文本的场景。原因之一就是他们是不可变的。另一个原因是不是所有字符串的值都能在编写代码时就知道:例如,如果想要获取用户输入并储存该怎么办呢?为此,Rust 有第二个字符串类型,`String`。这个类型储存在堆上所以能够储存在编译时未知大小的文本。可以用 `from` 函数从字符串字面值来创建 `String`,如下:
|
我们已经见过字符串字面值,字符串值被硬编码进程序里。字符串字面值是很方便的,不过他们并不适合使用文本的每一种场景。原因之一就是他们是不可变的。另一个原因是并不是所有字符串的值都能在编写代码时就知道:例如,要是想获取用户输入并存储该怎么办呢?为此,Rust 有第二个字符串类型,`String`。这个类型被分配到堆上,所以能够存储在编译时未知大小的文本。可以使用 `from` 函数基于字符串字面值来创建 `String`,如下:
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
let s = String::from("hello");
|
let s = String::from("hello");
|
||||||
```
|
```
|
||||||
|
|
||||||
这两个冒号(`::`)运算符允许将特定的 `from` 函数置于 `String` 类型的命名空间(namespace)下而不需要使用类似 `string_from` 这样的名字。在第五章的 “方法语法”(“Method Syntax”)部分会着重讲解这个语法而且在第七章会讲到模块的命名空间。
|
这两个冒号(`::`)是运算符,允许将特定的 `from` 函数置于 `String` 类型的命名空间(namespace)下,而不需要使用类似 `string_from` 这样的名字。在第五章的 “方法语法”(“Method Syntax”)部分会着重讲解这个语法而且在第七章的 “模块定义” 中会讲到模块的命名空间。
|
||||||
|
|
||||||
这类字符串 **可以** 被修改:
|
**可以** 修改此类字符串 :
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
let mut s = String::from("hello");
|
let mut s = String::from("hello");
|
||||||
|
|
||||||
s.push_str(", world!"); // push_str() appends a literal to a String
|
s.push_str(", world!"); // push_str() 在字符串后追加字面值
|
||||||
|
|
||||||
println!("{}", s); // This will print `hello, world!`
|
println!("{}", s); // 将打印 `hello, world!`
|
||||||
```
|
```
|
||||||
|
|
||||||
那么这里有什么区别呢?为什么 `String` 可变而字面值却不行呢?区别在于两个类型对内存的处理上。
|
那么这里有什么区别呢?为什么 `String` 可变而字面值却不行呢?区别在于两个类型对内存的处理上。
|
||||||
|
|
||||||
### 内存与分配
|
### 内存与分配
|
||||||
|
|
||||||
对于字符串字面值的情况,我们在编译时就知道其内容所以它直接被硬编码进最终的可执行文件中,这使得字符串字面值快速且高效。不过这些属性都只来源于其不可变性。不幸的是,我们不能为了每一个在编译时未知大小的文本而将一块内存放入二进制文件中而它的大小还可能随着程序运行而改变。
|
就字符串字面值来说,我们在编译时就知道其内容,所以文本被直接硬编码进最终的可执行文件中。这使得字符串字面值快速且高效。不过这些特性都只得益于字符串字面值的不可变性。不幸的是,我们不能为了每一个在编译时大小未知的文本而将一块内存放入二进制文件中,并且它的大小还可能随着程序运行而改变。
|
||||||
|
|
||||||
对于 `String` 类型,为了支持一个可变,可增长的文本片段,需要在堆上分配一块在编译时未知大小的内存来存放内容。这意味着:
|
对于 `String` 类型,为了支持一个可变,可增长的文本片段,需要在堆上分配一块在编译时未知大小的内存来存放内容。这意味着:
|
||||||
|
|
||||||
1. 内存必须在运行时向操作系统请求。
|
* 必须在运行时向操作系统请求内存。
|
||||||
2. 需要一个当我们处理完 `String` 时将内存返回给操作系统的方法。
|
* 需要一个当我们处理完 `String` 时将内存返回给操作系统的方法。
|
||||||
|
|
||||||
第一部分由我们完成:当调用 `String::from` 时,它的实现 (*implementation*) 请求其所需的内存。这在编程语言中是非常通用的。
|
第一部分由我们完成:当调用 `String::from` 时,它的实现 (*implementation*) 请求其所需的内存。这在编程语言中是非常通用的。
|
||||||
|
|
||||||
然而,第二部分实现起来就各有区别了。在有 **垃圾回收**(*garbage collector*,*GC*)的语言中, GC 记录并清除不再使用的内存,而我们作为程序员,并不需要关心他们。没有 GC 的话,识别出不再使用的内存并调用代码显式释放就是我们程序员的责任了,正如请求内存的时候一样。从历史的角度上说正确处理内存回收曾经是一个困难的编程问题。如果忘记回收了会浪费内存。如果过早回收了,将会出现无效变量。如果重复回收,这也是个 bug。我们需要 `allocate` 和 `free` 一一对应。
|
然而,第二部分实现起来就各有区别了。在有 **垃圾回收**(*garbage collector*,*GC*)的语言中, GC 记录并清除不再使用的内存,而我们并不需要关心它。没有 GC 的话,识别出不再使用的内存并调用代码显式释放就是我们的责任了,跟请求内存的时候一样。从历史的角度上说正确处理内存回收曾经是一个困难的编程问题。如果忘记回收了会浪费内存。如果过早回收了,将会出现无效变量。如果重复回收,这也是个 bug。我们需要精确的为一个 `allocate` 配对一个 `free`。
|
||||||
|
|
||||||
Rust 采取了一个不同的策略:内存在拥有它的变量离开作用域后就被自动释放。下面是示例 4-1 中作用域例子的一个使用 `String` 而不是字符串字面值的版本:
|
Rust 采取了一个不同的策略:内存在拥有它的变量离开作用域后就被自动释放。下面是示例 4-1 中作用域例子的一个使用 `String` 而不是字符串字面值的版本:
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
{
|
{
|
||||||
let s = String::from("hello"); // s is valid from this point forward
|
let s = String::from("hello"); // 从此处起,s 是有效的
|
||||||
|
|
||||||
// do stuff with s
|
// 使用 s
|
||||||
} // this scope is now over, and s is no
|
} // 此作用域已结束,
|
||||||
// longer valid
|
// s 不再有效
|
||||||
```
|
```
|
||||||
|
|
||||||
这里是一个将 `String` 需要的内存返回给操作系统的很自然的位置:当 `s` 离开作用域的时候。当变量离开作用域,Rust 为其调用一个特殊的函数。这个函数叫做 `drop`,在这里 `String` 的作者可以放置释放内存的代码。Rust 在结尾的 `}` 处自动调用 `drop`。
|
这是一个将 `String` 需要的内存返回给操作系统的很自然的位置:当 `s` 离开作用域的时候。当变量离开作用域,Rust 为我们调用一个特殊的函数。这个函数叫做 `drop`,在这里 `String` 的作者可以放置释放内存的代码。Rust 在结尾的 `}` 处自动调用 `drop`。
|
||||||
|
|
||||||
> 注意:在 C++ 中,这种 item 在生命周期结束时释放资源的方法有时被称作 **资源获取即初始化**(*Resource Acquisition Is Initialization (RAII)*)。如果你使用过 RAII 模式的话应该对 Rust 的 `drop` 函数并不陌生。
|
> 注意:在 C++ 中,这种 item 在生命周期结束时释放资源的模式有时被称作 **资源获取即初始化**(*Resource Acquisition Is Initialization (RAII)*)。如果你使用过 RAII 模式的话应该对 Rust 的 `drop` 函数并不陌生。
|
||||||
|
|
||||||
这个模式对编写 Rust 代码的方式有着深远的影响。现在它看起来很简单,不过在更复杂的场景下代码的行为可能是不可预测的,比如当有多个变量使用在堆上分配的内存时。现在让我们探索一些这样的场景。
|
这个模式对编写 Rust 代码的方式有着深远的影响。现在它看起来很简单,不过在更复杂的场景下代码的行为可能是不可预测的,比如当有多个变量使用在堆上分配的内存时。现在让我们探索一些这样的场景。
|
||||||
|
|
||||||
#### 变量与数据交互的方式(一):移动
|
#### 变量与数据交互的方式(一):移动
|
||||||
|
|
||||||
Rust 中的多个变量可以采用一种独特的方式与同一数据交互。让我们看看示例 4-2 中一个使用整型的例子:
|
Rust 中的多个变量可以采用一种独特的方式与同一数据交互。让我们看看示例 4-2 中一个使用整型的例子。
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
let x = 5;
|
let x = 5;
|
||||||
let y = x;
|
let y = x;
|
||||||
```
|
```
|
||||||
|
|
||||||
<span class="caption">示例 4-2:将变量 `x` 赋值给 `y`</span>
|
<span class="caption">示例 4-2:将变量 `x` 的整数值赋给 `y`</span>
|
||||||
|
|
||||||
根据其他语言的经验我们大致可以猜到这在干什么:“将 `5` 绑定到 `x`;接着生成一个值 `x` 的拷贝并绑定到 `y`”。现在有了两个变量,`x` 和 `y`,都等于 `5`。这也正是事实上发生了的,因为正数是有已知固定大小的简单值,所以这两个 `5` 被放入了栈中。
|
我们大致可以猜到这在干什么:“将 `5` 绑定到 `x`;接着生成一个值 `x` 的拷贝并绑定到 `y`”。现在有了两个变量,`x` 和 `y`,都等于 `5`。这也正是事实上发生了的,因为正数是有已知固定大小的简单值,所以这两个 `5` 被放入了栈中。
|
||||||
|
|
||||||
现在看看这个 `String` 版本:
|
现在看看这个 `String` 版本:
|
||||||
|
|
||||||
@ -149,29 +145,29 @@ let s2 = s1;
|
|||||||
|
|
||||||
这看起来与上面的代码非常类似,所以我们可能会假设他们的运行方式也是类似的:也就是说,第二行可能会生成一个 `s1` 的拷贝并绑定到 `s2` 上。不过,事实上并不完全是这样。
|
这看起来与上面的代码非常类似,所以我们可能会假设他们的运行方式也是类似的:也就是说,第二行可能会生成一个 `s1` 的拷贝并绑定到 `s2` 上。不过,事实上并不完全是这样。
|
||||||
|
|
||||||
为了更全面的解释这个问题,让我们看看图 4-1 中 `String` 真正是什么样的。`String` 由三部分组成,如图左侧所示:一个指向存放字符串内容内存的指针,一个长度,和一个容量。这一组数据储存在栈上。右侧则是堆上存放内容的内存部分。
|
看看图 4-1 以了解 `String` 的底层会发生什么。`String` 由三部分组成,如图左侧所示:一个指向存放字符串内容内存的指针,一个长度,和一个容量。这一组数据存储在栈上。右侧则是堆上存放内容的内存部分。
|
||||||
|
|
||||||
<img alt="String in memory" src="img/trpl04-01.svg" class="center" style="width: 50%;" />
|
<img alt="String in memory" src="img/trpl04-01.svg" class="center" style="width: 50%;" />
|
||||||
|
|
||||||
<span class="caption">图 4-1:一个绑定到 `s1` 的拥有值 `"hello"` 的 `String` 的内存表现</span>
|
<span class="caption">图 4-1:将值 `"hello"` 绑定给 `s1` 的 `String` 在内存中的表现形式</span>
|
||||||
|
|
||||||
长度代表当前 `String` 的内容使用了多少字节的内存。容量是 `String` 从操作系统总共获取了多少字节的内存。长度与容量的区别是很重要的,不过这在目前为止的场景中并不重要,所以可以暂时忽略容量。
|
长度表示 `String` 的内容当前使用了多少字节的内存。容量是 `String` 从操作系统总共获取了多少字节的内存。长度与容量的区别是很重要的,不过在当前上下文中并不重要,所以现在可以忽略容量。
|
||||||
|
|
||||||
当我们把 `s1` 赋值给 `s2`,`String` 的数据被复制了,这意味着我们从栈上拷贝了它的指针、长度和容量。我们并没有复制堆上指针所指向的数据。换句话说,内存中数据的表现如图 4-2 所示。
|
当我们将 `s1` 赋值给 `s2`,`String` 的数据被复制了,这意味着我们从栈上拷贝了它的指针、长度和容量。我们并没有复制指针指向的堆上数据。换句话说,内存中数据的表现如图 4-2 所示。
|
||||||
|
|
||||||
<img alt="s1 and s2 pointing to the same value" src="img/trpl04-02.svg" class="center" style="width: 50%;" />
|
<img alt="s1 and s2 pointing to the same value" src="img/trpl04-02.svg" class="center" style="width: 50%;" />
|
||||||
|
|
||||||
<span class="caption">图 4-2:变量 `s2` 的内存表现,它有一份 `s1` 指针、长度和容量的拷贝</span>
|
<span class="caption">图 4-2:变量 `s2` 的内存表现,它有一份 `s1` 指针、长度和容量的拷贝</span>
|
||||||
|
|
||||||
这个表现形式看起来 **并不像** 图 4-3 中的那样,但是如果 Rust 也拷贝了堆上的数据后内存看起来会是如何呢。如果 Rust 这么做了,那么操作 `s2 = s1` 在堆上数据比较大的时候可能会对运行时性能造成非常大的影响。
|
这个表现形式看起来 **并不像** 图 4-3 中的那样,如果 Rust 也拷贝了堆上的数据,那么内存看起来就是这样的。如果 Rust 这么做了,那么操作 `s2 = s1` 在堆上数据比较大的时候会对运行时性能造成非常大的影响。
|
||||||
|
|
||||||
<img alt="s1 and s2 to two places" src="img/trpl04-03.svg" class="center" style="width: 50%;" />
|
<img alt="s1 and s2 to two places" src="img/trpl04-03.svg" class="center" style="width: 50%;" />
|
||||||
|
|
||||||
<span class="caption">图 4-3:另一个 `s2 = s1` 时可能的内存表现,如果 Rust 同时也拷贝了堆上的数据的话</span>
|
<span class="caption">图 4-3:另一个 `s2 = s1` 时可能的内存表现,如果 Rust 同时也拷贝了堆上的数据的话</span>
|
||||||
|
|
||||||
之前,我们提到过当变量离开作用域后 Rust 自动调用 `drop` 函数并清理变量的堆内存。不过图 4-4 展示了两个数据指针指向了同一位置。这就有了一个问题:当 `s2` 和 `s1` 离开作用域,他们都会尝试释放相同的内存。这是一个叫做 **二次释放**(*double free*)的错误,也是之前提到过的内存安全性 bug 之一。两次释放(相同)内存会导致内存污染,它可能会导致潜在的安全漏洞。
|
之前,我们提到过当变量离开作用域后,Rust 自动调用 `drop` 函数并清理变量的堆内存。不过图 4-2 展示了两个数据指针指向了同一位置。这就有了一个问题:当 `s2` 和 `s1` 离开作用域,他们都会尝试释放相同的内存。这是一个叫做 **二次释放**(*double free*)的错误,也是之前提到过的内存安全性 bug 之一。两次释放(相同)内存会导致内存污染,它可能会导致潜在的安全漏洞。
|
||||||
|
|
||||||
为了确保内存安全,这种场景下 Rust 的处理有另一个细节值得注意。与其尝试拷贝被分配的内存,Rust 则认为 `s1` 不再有效,因此 Rust 不需要在 `s1` 离开作用域后清理任何东西。看看在 `s2` 被创建之后尝试使用 `s1` 会发生什么:
|
为了确保内存安全,这种场景下 Rust 的处理有另一个细节值得注意。与其尝试拷贝被分配的内存,Rust 则认为 `s1` 不再有效,因此 Rust 不需要在 `s1` 离开作用域后清理任何东西。看看在 `s2` 被创建之后尝试使用 `s1` 会发生什么;它不会工作:
|
||||||
|
|
||||||
```rust,ignore
|
```rust,ignore
|
||||||
let s1 = String::from("hello");
|
let s1 = String::from("hello");
|
||||||
@ -196,13 +192,13 @@ error[E0382]: use of moved value: `s1`
|
|||||||
not implement the `Copy` trait
|
not implement the `Copy` trait
|
||||||
```
|
```
|
||||||
|
|
||||||
如果你在其他语言中听说过术语 “浅拷贝”(“shallow copy”)和 “深拷贝”(“deep copy”),那么拷贝指针、长度和容量而不拷贝数据可能听起来像浅拷贝。不过因为 Rust 同时使第一个变量无效化了,这个操作被称为 **移动**(*move*),而不是浅拷贝。上面的例子可以解读为 `s1` 被 **移动** 到了 `s2` 中。那么具体发生了什么,如图 4-4 所示。
|
如果你在其他语言中听说过术语 **浅拷贝**(*shallow copy*)和 **深拷贝**(*deep copy*),那么拷贝指针、长度和容量而不拷贝数据可能听起来像浅拷贝。不过因为 Rust 同时使第一个变量无效了,这个操作被称为 **移动**(*move*),而不是浅拷贝。上面的例子可以解读为 `s1` 被 **移动** 到了 `s2` 中。那么具体发生了什么,如图 4-4 所示。
|
||||||
|
|
||||||
<img alt="s1 moved to s2" src="img/trpl04-04.svg" class="center" style="width: 50%;" />
|
<img alt="s1 moved to s2" src="img/trpl04-04.svg" class="center" style="width: 50%;" />
|
||||||
|
|
||||||
<span class="caption">图 4-4:`s1` 无效化之后的内存表现</span>
|
<span class="caption">图 4-4:`s1` 无效之后的内存表现</span>
|
||||||
|
|
||||||
这样就解决了我们的麻烦!因为只有 `s2` 是有效的,当其离开作用域,它就释放自己的内存,完毕。
|
这样就解决了我们的问题!因为只有 `s2` 是有效的,当其离开作用域,它就释放自己的内存,完毕。
|
||||||
|
|
||||||
另外,这里还隐含了一个设计选择:Rust 永远也不会自动创建数据的 “深拷贝”。因此,任何 **自动** 的复制可以被认为对运行时性能影响较小。
|
另外,这里还隐含了一个设计选择:Rust 永远也不会自动创建数据的 “深拷贝”。因此,任何 **自动** 的复制可以被认为对运行时性能影响较小。
|
||||||
|
|
||||||
@ -219,13 +215,13 @@ let s2 = s1.clone();
|
|||||||
println!("s1 = {}, s2 = {}", s1, s2);
|
println!("s1 = {}, s2 = {}", s1, s2);
|
||||||
```
|
```
|
||||||
|
|
||||||
这段代码能正常运行,也是如何显式产生图 4-3 中行为的方式,这里堆上的数据 **确实** 被复制了。
|
这段代码能正常运行,并且明确产生图 4-3 中行为,这里堆上的数据 **确实** 被复制了。
|
||||||
|
|
||||||
当出现 `clone` 调用时,你知道一些特定的代码被执行而且这些代码可能相当消耗资源。你很容易察觉到一些不寻常的事情正在发生。
|
当出现 `clone` 调用时,你知道一些特定的代码被执行而且这些代码可能相当消耗资源。你很容易察觉到一些不寻常的事情正在发生。
|
||||||
|
|
||||||
#### 只在栈上的数据:拷贝
|
#### 只在栈上的数据:拷贝
|
||||||
|
|
||||||
这里还有一个没有提到的小窍门。这些代码使用了整型并且是有效的,他们是之前示例 4-2 中的一部分:
|
这里还有一个没有提到的小窍门。这些代码使用了整型并且是有效的,他们是示例 4-2 中的一部分:
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
let x = 5;
|
let x = 5;
|
||||||
@ -234,98 +230,95 @@ let y = x;
|
|||||||
println!("x = {}, y = {}", x, y);
|
println!("x = {}, y = {}", x, y);
|
||||||
```
|
```
|
||||||
|
|
||||||
他们似乎与我们刚刚学到的内容相抵触:没有调用 `clone`,不过 `x` 依然有效且没有被移动到 `y` 中。
|
但这段代码似乎与我们刚刚学到的内容相矛盾:没有调用 `clone`,不过 `x` 依然有效且没有被移动到 `y` 中。
|
||||||
|
|
||||||
原因是像整型这样的在编译时已知大小的类型被整个储存在栈上,所以拷贝其实际的值是快速的。这意味着没有理由在创建变量 `y` 后使 `x` 无效。换句话说,这里没有深浅拷贝的区别,所以这里调用 `clone` 并不会与通常的浅拷贝有什么不同,我们可以不用管它。
|
原因是像整型这样的在编译时已知大小的类型被整个存储在栈上,所以拷贝其实际的值是快速的。这意味着没有理由在创建变量 `y` 后使 `x` 无效。换句话说,这里没有深浅拷贝的区别,所以这里调用 `clone` 并不会与通常的浅拷贝有什么不同,我们可以不用管它。
|
||||||
|
|
||||||
Rust 有一个叫做 `Copy` trait 的特殊注解,可以用在类似整型这样的储存在栈上的类型(第十章详细讲解 trait)。如果一个类型拥有 `Copy` trait,一个旧的变量在将其赋值给其他变量后仍然可用。Rust 不允许自身或其任何部分实现了 `Drop` trait 的类型使用 `Copy` trait。如果我们对其值离开作用域时需要特殊处理的类型使用 `Copy` 注解,将会出现一个编译时错误。关于如何为你的类型增加 `Copy` 注解,请阅读附录 C 中的可导出 trait。
|
Rust 有一个叫做 `Copy` trait 的特殊注解,可以用在类似整型这样的存储在栈上的类型上(第十章详细讲解 trait)。如果一个类型拥有 `Copy` trait,一个旧的变量在将其赋值给其他变量后仍然可用。Rust 不允许自身或其任何部分实现了 `Drop` trait 的类型使用 `Copy` trait。如果我们对其值离开作用域时需要特殊处理的类型使用 `Copy` 注解,将会出现一个编译时错误。要学习如何为你的类型增加 `Copy` 注解,请阅读附录 C 中的 “可派生的 trait”。
|
||||||
|
|
||||||
那么什么类型是 `Copy` 的呢?可以查看给定类型的文档来确认,不过作为一个通用的规则,任何简单标量值的组合可以是 `Copy` 的,任何需要分配内存,或者本身就是某种形式资源的类型不会是 `Copy` 的。如下是一些 `Copy` 的类型:
|
那么什么类型是 `Copy` 的呢?可以查看给定类型的文档来确认,不过作为一个通用的规则,任何简单标量值的组合可以是 `Copy` 的,不需要分配内存或某种形式资源的类型是 `Copy` 的。如下是一些 `Copy` 的类型:
|
||||||
|
|
||||||
* 所有整数类型,比如 `u32`。
|
* 所有整数类型,比如 `u32`。
|
||||||
* 布尔类型,`bool`,它的值是 `true` 和 `false`。
|
* 布尔类型,`bool`,它的值是 `true` 和 `false`。
|
||||||
* 所有浮点数类型,比如 `f64`。
|
* 所有浮点数类型,比如 `f64`。
|
||||||
* 元组,当且仅当其包含的类型也都是 `Copy` 的时候。`(i32, i32)` 是 `Copy` 的,不过 `(i32, String)` 就不是。
|
* 字符类型,`char`。
|
||||||
|
* 元组,当且仅当其包含的类型也都是 `Copy` 的时候。比如,`(i32, i32)` 是 `Copy` 的,但 `(i32, String)` 就不是。
|
||||||
|
|
||||||
### 所有权与函数
|
### 所有权与函数
|
||||||
|
|
||||||
将值传递给函数在语义上与给变量赋值相似。向函数传递值可能会移动或者复制,就像赋值语句一样。示例 4-7 是一个展示变量何时进入和离开作用域的例子:
|
将值传递给函数在语义上与给变量赋值相似。向函数传递值可能会移动或者复制,就像赋值语句一样。示例 4-3 使用注释展示变量何时进入和离开作用域:
|
||||||
|
|
||||||
<span class="filename">文件名: src/main.rs</span>
|
<span class="filename">文件名: src/main.rs</span>
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
fn main() {
|
fn main() {
|
||||||
let s = String::from("hello"); // s comes into scope.
|
let s = String::from("hello"); // s 进入作用域
|
||||||
|
|
||||||
takes_ownership(s); // s's value moves into the function...
|
takes_ownership(s); // s 的值移动到函数里...
|
||||||
// ... and so is no longer valid here.
|
// ... 所以到这里不再有效
|
||||||
|
|
||||||
let x = 5; // x comes into scope.
|
let x = 5; // x 进入作用域
|
||||||
|
|
||||||
makes_copy(x); // x would move into the function,
|
makes_copy(x); // x 应该移动函数里,
|
||||||
// but i32 is Copy, so it’s okay to still
|
// 但 i32 是 Copy 的,所以在后面可继续使用 x
|
||||||
// use x afterward.
|
|
||||||
|
|
||||||
} // Here, x goes out of scope, then s. But since s's value was moved, nothing
|
} // 这里, x 先移出了作用域,然后是 s。但因为 s 的值已被移走,
|
||||||
// special happens.
|
// 所以不会有特殊操作
|
||||||
|
|
||||||
fn takes_ownership(some_string: String) { // some_string comes into scope.
|
fn takes_ownership(some_string: String) { // some_string 进入作用域
|
||||||
println!("{}", some_string);
|
println!("{}", some_string);
|
||||||
} // Here, some_string goes out of scope and `drop` is called. The backing
|
} // 这里,some_string 移出作用域并调用 `drop` 方法。占用的内存被释放
|
||||||
// memory is freed.
|
|
||||||
|
|
||||||
fn makes_copy(some_integer: i32) { // some_integer comes into scope.
|
fn makes_copy(some_integer: i32) { // some_integer 进入作用域
|
||||||
println!("{}", some_integer);
|
println!("{}", some_integer);
|
||||||
} // Here, some_integer goes out of scope. Nothing special happens.
|
} // 这里,some_integer 移出作用域。不会有特殊操作
|
||||||
```
|
```
|
||||||
|
|
||||||
<span class="caption">示例 4-3:带有所有权和作用域标注的函数</span>
|
<span class="caption">示例 4-3:带有所有权和作用域注释的函数</span>
|
||||||
|
|
||||||
当尝试在调用 `takes_ownership` 后使用 `s` 时,Rust 会抛出一个编译时错误。这些静态检查使我们免于犯错。试试在 `main` 函数中添加使用 `s` 和 `x` 的代码来看看哪里能使用他们,以及所有权规则会在哪里阻止我们这么做。
|
当尝试在调用 `takes_ownership` 后使用 `s` 时,Rust 会抛出一个编译时错误。这些静态检查使我们免于犯错。试试在 `main` 函数中添加使用 `s` 和 `x` 的代码来看看哪里能使用他们,以及所有权规则会在哪里阻止我们这么做。
|
||||||
|
|
||||||
### 返回值与作用域
|
### 返回值与作用域
|
||||||
|
|
||||||
返回值也可以转移作用域。这里是一个拥有与示例 4-3 中类似标注的例子:
|
返回值也可以转移所有权。示例 4-4 与示例 4-3 一样带有类似的注释。
|
||||||
|
|
||||||
<span class="filename">文件名: src/main.rs</span>
|
<span class="filename">文件名: src/main.rs</span>
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
fn main() {
|
fn main() {
|
||||||
let s1 = gives_ownership(); // gives_ownership moves its return
|
let s1 = gives_ownership(); // gives_ownership 将返回值
|
||||||
// value into s1.
|
// 移给 s1
|
||||||
|
|
||||||
let s2 = String::from("hello"); // s2 comes into scope.
|
let s2 = String::from("hello"); // s2 进入作用域
|
||||||
|
|
||||||
let s3 = takes_and_gives_back(s2); // s2 is moved into
|
let s3 = takes_and_gives_back(s2); // s2 被移动到
|
||||||
// takes_and_gives_back, which also
|
// takes_and_gives_back 中,
|
||||||
// moves its return value into s3.
|
// 它也将返回值移给 s3
|
||||||
} // Here, s3 goes out of scope and is dropped. s2 goes out of scope but was
|
} // 这里, s3 移出作用域并被丢弃。s2 也移出作用域,但已被移走,
|
||||||
// moved, so nothing happens. s1 goes out of scope and is dropped.
|
// 所以什么也不会发生。s1 移出作用域并被丢弃
|
||||||
|
|
||||||
fn gives_ownership() -> String { // gives_ownership will move its
|
fn gives_ownership() -> String { // gives_ownership 将返回值移动给
|
||||||
// return value into the function
|
// 调用它的函数
|
||||||
// that calls it.
|
|
||||||
|
|
||||||
let some_string = String::from("hello"); // some_string comes into scope.
|
let some_string = String::from("hello"); // some_string 进入作用域.
|
||||||
|
|
||||||
some_string // some_string is returned and
|
some_string // 返回 some_string 并移出给调用的函数
|
||||||
// moves out to the calling
|
|
||||||
// function.
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// takes_and_gives_back will take a String and return one.
|
// takes_and_gives_back 将传入字符串并返回该值
|
||||||
fn takes_and_gives_back(a_string: String) -> String { // a_string comes into
|
fn takes_and_gives_back(a_string: String) -> String { // a_string 进入作用域
|
||||||
// scope.
|
|
||||||
|
|
||||||
a_string // a_string is returned and moves out to the calling function.
|
a_string // 返回 a_string 并移出给调用的函数
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
变量的所有权总是遵循相同的模式:将值赋值给另一个变量时移动它。当持有堆中数据值的变量离开作用域时,其值将通过 `drop` 被清理掉,除非数据被移动为另一个变量所有。
|
<span class="caption">示例 4-4: 转移返回值的所有权</span>
|
||||||
|
|
||||||
在每一个函数中都获取并接着返回所有权可能有些冗余。如果我们想要函数使用一个值但不获取所有权该怎么办呢?如果我们还要接着使用它的话,每次都传递出去再传回来就有点烦人了,如果真的这样做,有时一个函数就需要有多个返回值。
|
变量的所有权总是遵循相同的模式:将值赋给另一个变量时移动它。当持有堆中数据值的变量离开作用域时,其值将通过 `drop` 被清理掉,除非数据被移动为另一个变量所有。
|
||||||
|
|
||||||
当然,我们可以使用元组来返回多个值,像这样:
|
在每一个函数中都获取所有权并接着返回所有权有些啰嗦。如果我们想要函数使用一个值但不获取所有权该怎么办呢?如果我们还要接着使用它的话,每次都传进去再返回来就有点烦人了,除此之外,我们也可能想返回函数体中产生的一些数据。
|
||||||
|
|
||||||
|
我们可以使用元组来返回多个值,如示例 4-5 所示。
|
||||||
|
|
||||||
<span class="filename">文件名: src/main.rs</span>
|
<span class="filename">文件名: src/main.rs</span>
|
||||||
|
|
||||||
@ -339,10 +332,12 @@ fn main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn calculate_length(s: String) -> (String, usize) {
|
fn calculate_length(s: String) -> (String, usize) {
|
||||||
let length = s.len(); // len() returns the length of a String.
|
let length = s.len(); // len() 返回字符串的长度
|
||||||
|
|
||||||
(s, length)
|
(s, length)
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
<span class="caption">示例 4-5: 返回参数的所有权</span>
|
||||||
|
|
||||||
但是这未免有些形式主义,而且这种场景应该很常见。幸运的是,Rust 对此提供了一个功能,叫做 **引用**(*references*)。
|
但是这未免有些形式主义,而且这种场景应该很常见。幸运的是,Rust 对此提供了一个功能,叫做 **引用**(*references*)。
|
||||||
|
Loading…
Reference in New Issue
Block a user