所有権の移動
2022/2/17 22:02:00
Copy
特質を持たない変数をコピーすると、所有権の移動が発生し、コピー元の変数は使用が不可能になる。
let s1 = String::from("hello");
let s2 = s1;
println!("{}", s1); //error
/*
error[E0382]: borrow of moved value: `s1`
--> src\main.rs:5:16
|
3 | let s1 = String::from("hello");
| -- move occurs because `s1` has type `String`, which does not implement the `Copy` trait
4 | let s2 = s1;
| -- value moved here
5 | println!("{}", s1); //hello
| ^^ value borrowed here after move
*/
Copy
特質を持たない変数を関数に渡すと、所有権の移動が発生し、関数に渡した変数は使用不可能になる。
fn main() {
let s = String::from("hello");
test(s);
println!("{}", s); //error
/*
error[E0382]: borrow of moved value: `s`
--> src\main.rs:4:20
|
2 | let s = String::from("hello");
| - move occurs because `s` has type `String`, which does not implement the `Copy` trait
3 | test(s);
| - value moved here
4 | println!("{}", s); //error
| ^ value borrowed here after move
*/
}
fn test(s:String) {
println!("{}", s); //hello
}
ベクタの値を走査する時にベクタに参照(&
)を使う理由は、所有権の移動が発生し、走査して以降ベクタが利用できなくなるからである。
参照を使わずに所有権の移動する例
let v = vec![100, 32, 57];
for i in v {
println!("{}", i);
}
for i in v {
println!("{}", i);
}
error[E0382]: use of moved value: `v`
--> src\main.rs:8:10
|
3 | let v = vec![100, 32, 57];
| - move occurs because `v` has type `Vec<i32>`, which does not implement the `Copy` trait
4 | for i in v {
| -
| |
| `v` moved due to this implicit call to `.into_iter()`
| help: consider borrowing to avoid moving into the for loop: `&v`
...
8 | for i in v {
| ^ value used here after move
|
Rustでは関数で作成したデータの参照を戻り値で渡すことはできないので、データ(String)を直接返せばよい。戻り値による所有権の移動が発生して無事データを渡すことができる。
fn main() {
let s = no_dangle();
println!("{}", s); //hello
}
fn no_dangle() -> String {
let s = String::from("hello");
s
}
関数に変数を渡すことによって所有権の移動が発生するが、戻り値によっても所有権の移動が発生する。
fn main() {
let s1 = String::from("hello");
let s2 = test(s1);
println!("{}", s2); //hello
}
fn test(s:String) -> String {
s
}
関数に変数を渡すことによる値の移動が起きて欲しくない時がある。そういった時には参照渡しを利用する。参照渡しでは所有権の移動は発生しない。
fn main() {
let s1 = String::from("hello");
let len = calculate_length(&s1);
println!("'{}'の長さは、{}です", s1, len); //'hello'の長さは、5です
}
fn calculate_length(s: &String) -> usize {
s.len()
}