2015-01-16 15:30:27 -05:00
|
|
|
|
% Ownership
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
This guide is one of three presenting Rust’s ownership system. This is one of
|
|
|
|
|
Rust’s most unique and compelling features, with which Rust developers should
|
|
|
|
|
become quite acquainted. Ownership is how Rust achieves its largest goal,
|
2015-05-08 00:42:10 +02:00
|
|
|
|
memory safety. There are a few distinct concepts, each with its own
|
2015-04-24 16:43:36 -04:00
|
|
|
|
chapter:
|
|
|
|
|
|
2015-05-08 22:15:14 +02:00
|
|
|
|
* ownership, which you’re reading now
|
2015-04-24 16:43:36 -04:00
|
|
|
|
* [borrowing][borrowing], and their associated feature ‘references’
|
|
|
|
|
* [lifetimes][lifetimes], an advanced concept of borrowing
|
|
|
|
|
|
|
|
|
|
These three chapters are related, and in order. You’ll need all three to fully
|
|
|
|
|
understand the ownership system.
|
|
|
|
|
|
|
|
|
|
[borrowing]: references-and-borrowing.html
|
|
|
|
|
[lifetimes]: lifetimes.html
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
|
|
|
|
# Meta
|
|
|
|
|
|
|
|
|
|
Before we get to the details, two important notes about the ownership system.
|
|
|
|
|
|
|
|
|
|
Rust has a focus on safety and speed. It accomplishes these goals through many
|
2015-04-24 16:43:36 -04:00
|
|
|
|
‘zero-cost abstractions’, which means that in Rust, abstractions cost as little
|
2014-11-04 02:52:36 -05:00
|
|
|
|
as possible in order to make them work. The ownership system is a prime example
|
2015-05-08 22:15:14 +02:00
|
|
|
|
of a zero-cost abstraction. All of the analysis we’ll talk about in this guide
|
2014-11-04 02:52:36 -05:00
|
|
|
|
is _done at compile time_. You do not pay any run-time cost for any of these
|
|
|
|
|
features.
|
|
|
|
|
|
|
|
|
|
However, this system does have a certain cost: learning curve. Many new users
|
2015-04-24 16:43:36 -04:00
|
|
|
|
to Rust experience something we like to call ‘fighting with the borrow
|
|
|
|
|
checker’, where the Rust compiler refuses to compile a program that the author
|
|
|
|
|
thinks is valid. This often happens because the programmer’s mental model of
|
|
|
|
|
how ownership should work doesn’t match the actual rules that Rust implements.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
You probably will experience similar things at first. There is good news,
|
|
|
|
|
however: more experienced Rust developers report that once they work with the
|
|
|
|
|
rules of the ownership system for a period of time, they fight the borrow
|
|
|
|
|
checker less and less.
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
With that in mind, let’s learn about ownership.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
|
|
|
|
# Ownership
|
|
|
|
|
|
2015-05-08 22:15:14 +02:00
|
|
|
|
[Variable bindings][bindings] have a property in Rust: they ‘have ownership’
|
2015-10-19 11:16:31 -04:00
|
|
|
|
of what they’re bound to. This means that when a binding goes out of scope,
|
2015-07-13 17:22:08 +02:00
|
|
|
|
Rust will free the bound resources. For example:
|
2012-09-15 17:09:21 -07:00
|
|
|
|
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```rust
|
2015-04-24 16:43:36 -04:00
|
|
|
|
fn foo() {
|
|
|
|
|
let v = vec![1, 2, 3];
|
2012-09-15 17:09:21 -07:00
|
|
|
|
}
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
When `v` comes into scope, a new [`Vec<T>`][vect] is created. In this case, the
|
|
|
|
|
vector also allocates space on [the heap][heap], for the three elements. When
|
|
|
|
|
`v` goes out of scope at the end of `foo()`, Rust will clean up everything
|
|
|
|
|
related to the vector, even the heap-allocated memory. This happens
|
|
|
|
|
deterministically, at the end of the scope.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
[vect]: ../std/vec/struct.Vec.html
|
|
|
|
|
[heap]: the-stack-and-the-heap.html
|
2015-05-08 00:42:10 +02:00
|
|
|
|
[bindings]: variable-bindings.html
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
# Move semantics
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
There’s some more subtlety here, though: Rust ensures that there is _exactly
|
|
|
|
|
one_ binding to any given resource. For example, if we have a vector, we can
|
|
|
|
|
assign it to another binding:
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
|
|
|
|
```rust
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v = vec![1, 2, 3];
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v2 = v;
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
But, if we try to use `v` afterwards, we get an error:
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
```rust,ignore
|
|
|
|
|
let v = vec![1, 2, 3];
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v2 = v;
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
println!("v[0] is: {}", v[0]);
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
It looks like this:
|
2015-01-12 13:48:59 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
```text
|
|
|
|
|
error: use of moved value: `v`
|
|
|
|
|
println!("v[0] is: {}", v[0]);
|
|
|
|
|
^
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
A similar thing happens if we define a function which takes ownership, and
|
|
|
|
|
try to use something after we’ve passed it as an argument:
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
```rust,ignore
|
|
|
|
|
fn take(v: Vec<i32>) {
|
|
|
|
|
// what happens here isn’t important.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
}
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v = vec![1, 2, 3];
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
take(v);
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
println!("v[0] is: {}", v[0]);
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-05-08 22:15:14 +02:00
|
|
|
|
Same error: ‘use of moved value’. When we transfer ownership to something else,
|
2015-04-24 16:43:36 -04:00
|
|
|
|
we say that we’ve ‘moved’ the thing we refer to. You don’t need some sort of
|
|
|
|
|
special annotation here, it’s the default thing that Rust does.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
## The details
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
The reason that we cannot use a binding after we’ve moved it is subtle, but
|
|
|
|
|
important. When we write code like this:
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
|
|
|
|
```rust
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v = vec![1, 2, 3];
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v2 = v;
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-05-08 22:15:14 +02:00
|
|
|
|
The first line allocates memory for the vector object, `v`, and for the data it
|
|
|
|
|
contains. The vector object is stored on the [stack][sh] and contains a pointer
|
|
|
|
|
to the content (`[1, 2, 3]`) stored on the [heap][sh]. When we move `v` to `v2`,
|
|
|
|
|
it creates a copy of that pointer, for `v2`. Which means that there would be two
|
|
|
|
|
pointers to the content of the vector on the heap. It would violate Rust’s
|
|
|
|
|
safety guarantees by introducing a data race. Therefore, Rust forbids using `v`
|
|
|
|
|
after we’ve done the move.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
[sh]: the-stack-and-the-heap.html
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
It’s also important to note that optimizations may remove the actual copy of
|
2015-05-08 22:15:14 +02:00
|
|
|
|
the bytes on the stack, depending on circumstances. So it may not be as
|
|
|
|
|
inefficient as it initially seems.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
## `Copy` types
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
We’ve established that when ownership is transferred to another binding, you
|
|
|
|
|
cannot use the original binding. However, there’s a [trait][traits] that changes this
|
|
|
|
|
behavior, and it’s called `Copy`. We haven’t discussed traits yet, but for now,
|
|
|
|
|
you can think of them as an annotation to a particular type that adds extra
|
|
|
|
|
behavior. For example:
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
|
|
|
|
```rust
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v = 1;
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v2 = v;
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
println!("v is: {}", v);
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
In this case, `v` is an `i32`, which implements the `Copy` trait. This means
|
|
|
|
|
that, just like a move, when we assign `v` to `v2`, a copy of the data is made.
|
|
|
|
|
But, unlike a move, we can still use `v` afterward. This is because an `i32`
|
|
|
|
|
has no pointers to data somewhere else, copying it is a full copy.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-05-30 11:51:25 +02:00
|
|
|
|
All primitive types implement the `Copy` trait and their ownership is
|
2015-06-08 18:01:40 +02:00
|
|
|
|
therefore not moved like one would assume, following the ´ownership rules´.
|
2015-10-19 11:16:31 -04:00
|
|
|
|
To give an example, the two following snippets of code only compile because the
|
|
|
|
|
`i32` and `bool` types implement the `Copy` trait.
|
2015-05-30 11:51:25 +02:00
|
|
|
|
|
|
|
|
|
```rust
|
|
|
|
|
fn main() {
|
|
|
|
|
let a = 5;
|
|
|
|
|
|
|
|
|
|
let _y = double(a);
|
|
|
|
|
println!("{}", a);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn double(x: i32) -> i32 {
|
|
|
|
|
x * 2
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
```rust
|
|
|
|
|
fn main() {
|
|
|
|
|
let a = true;
|
|
|
|
|
|
|
|
|
|
let _y = change_truth(a);
|
|
|
|
|
println!("{}", a);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn change_truth(x: bool) -> bool {
|
|
|
|
|
!x
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2015-11-17 02:39:09 +00:00
|
|
|
|
If we had used types that do not implement the `Copy` trait,
|
2015-05-30 11:51:25 +02:00
|
|
|
|
we would have gotten a compile error because we tried to use a moved value.
|
|
|
|
|
|
|
|
|
|
```text
|
|
|
|
|
error: use of moved value: `a`
|
|
|
|
|
println!("{}", a);
|
|
|
|
|
^
|
|
|
|
|
```
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
We will discuss how to make your own types `Copy` in the [traits][traits]
|
|
|
|
|
section.
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
[traits]: traits.html
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
# More than ownership
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
Of course, if we had to hand ownership back with every function we wrote:
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
|
|
|
|
```rust
|
2015-04-24 16:43:36 -04:00
|
|
|
|
fn foo(v: Vec<i32>) -> Vec<i32> {
|
|
|
|
|
// do stuff with v
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
// hand back ownership
|
|
|
|
|
v
|
2014-11-04 02:52:36 -05:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2015-05-07 21:31:10 +02:00
|
|
|
|
This would get very tedious. It gets worse the more things we want to take ownership of:
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
|
|
|
|
```rust
|
2015-04-24 16:43:36 -04:00
|
|
|
|
fn foo(v1: Vec<i32>, v2: Vec<i32>) -> (Vec<i32>, Vec<i32>, i32) {
|
|
|
|
|
// do stuff with v1 and v2
|
2014-11-04 02:52:36 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
// hand back ownership, and the result of our function
|
|
|
|
|
(v1, v2, 42)
|
2014-11-04 02:52:36 -05:00
|
|
|
|
}
|
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let v1 = vec![1, 2, 3];
|
|
|
|
|
let v2 = vec![1, 2, 3];
|
2012-09-15 17:09:21 -07:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
let (v1, v2, answer) = foo(v1, v2);
|
2014-11-04 02:52:36 -05:00
|
|
|
|
```
|
2014-03-23 16:05:01 -04:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
Ugh! The return type, return line, and calling the function gets way more
|
|
|
|
|
complicated.
|
2014-12-11 11:37:20 -05:00
|
|
|
|
|
2015-04-24 16:43:36 -04:00
|
|
|
|
Luckily, Rust offers a feature, borrowing, which helps us solve this problem.
|
|
|
|
|
It’s the topic of the next section!
|
2014-12-11 11:37:20 -05:00
|
|
|
|
|