質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
Rust

Rustは、MoFoが支援するプログラミング言語。高速性を維持しつつも、メモリ管理を安全に行うことが可能な言語です。同じコンパイル言語であるC言語やC++では困難だったマルチスレッドを実装しやすく、並行性という点においても優れています。

Q&A

解決済

1回答

615閲覧

?演算子を使うと可変借用が帰ってこない

Watching

総合スコア56

Rust

Rustは、MoFoが支援するプログラミング言語。高速性を維持しつつも、メモリ管理を安全に行うことが可能な言語です。同じコンパイル言語であるC言語やC++では困難だったマルチスレッドを実装しやすく、並行性という点においても優れています。

1グッド

1クリップ

投稿2021/09/25 15:08

編集2021/09/25 15:23

意味の無いコードですがこんな感じの問題で困っています

rust

1struct Element{ 2 element:i32 3} 4impl Element{ 5 pub fn set(&mut self,num:i32)->Result<(),&str>{ 6 if num=0 {return Err("えらー")} 7 self.element=num; 8 Ok(()) 9 } 10} 11 12struct Field{ 13 field:[Element;10] 14} 15impl Field{ 16 pub fn func(&mut self)->Result<(),&str>{ 17 self.field[0].set(0)?;//この処理が終わったら可変借用は返ってくるはずでは? 18 self.field[0].set(1)?;//second mutable borrow occurs here 19 Ok(()) 20 } 21}

?演算子を使わなければエラーが出ないので、?演算子を使うことで借用期間が延長されてしまっているのだと思いますがなぜそんなことが起きるのか分かりません。

sutonea👍を押しています

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

この事象を理解するためには、ライフタイム省略規則について知っておく必要があります.

詳細は The Rust Programming Language 日本語版 10.3 ライフタイムで参照を検証する の「ライフタイム省略」をご自身で読んで確認していただきたいですが、重要な部分だけ引用します.

最初の規則は、参照である各引数は、独自のライフタイム引数を得るというものです。換言すれば、 1引数の関数は、1つのライフタイム引数を得るということです: fn foo<'a>(x: &'a i32); 2つ引数のある関数は、2つの個別のライフタイム引数を得ます: fn foo<'a, 'b>(x: &'a i32, y: &'b i32); 以下同様。

2番目の規則は、1つだけ入力ライフタイム引数があるなら、そのライフタイムが全ての出力ライフタイム引数に代入されるというものです: fn foo<'a>(x: &'a i32) -> &'a i32。

3番目の規則は、複数の入力ライフタイム引数があるけれども、メソッドなのでそのうちの一つが&selfや&mut selfだったら、 selfのライフタイムが全出力ライフタイム引数に代入されるというものです。 この3番目の規則により、必要なシンボルの数が減るので、メソッドが遥かに読み書きしやすくなります。

今回の例で言えば、set関数は、1番目の規則によって以下のようになり、

rust

1pub fn set<'a>(&'a mut self, num: i32)-> Result<(), &str> {}

さらに2番目の規則によって、以下のようなライフタイムを持った関数だと解釈されます.

rust

1pub fn set<'a>(&'a mut self, num: i32)-> Result<(), &'a str> {}

func関数も同様に考えてみましょう.一旦疑似コードで表してみます.

rust

1// 疑似コードなのでコンパイルできません 2pub fn func<'a>(&'a mut self) -> Result<(), &'a str> { 3 Element::set::<'b>(&'b mut self.field[0], 0)?; 4 Element::set::<'c>(&'c mut self.field[0], 1)?; 5 Ok(()) 6}

ここで、&mut selfの制約からは、'b,'c'aより決して長くならないことが言えます.逆に、&strの制約からは、'b,'c'aより長くなる必要があることが言えます.そしてこれらを合わせると'a='b='cとなります.しかしこれでは、同じライフタイムを持った可変参照が同時に存在するので、借用規則に反しておりコンパイルエラーになるというわけです.

根本的な原因は、返り値のライフタイムが不当に狭くなっていることなので、今回であれば以下のようにライフタイムを'staticにすれば、コンパイルエラーは解消します.

rust

1struct Element { 2 element: i32, 3} 4impl Element { 5 pub fn set(&mut self, num: i32) -> Result<(), &'static str> { 6 if num == 0 { 7 return Err("えらー"); 8 } 9 self.element = num; 10 Ok(()) 11 } 12} 13 14struct Field { 15 field: [Element; 10], 16} 17impl Field { 18 pub fn func(&mut self) -> Result<(), &'static str> { 19 self.field[0].set(1)?; 20 self.field[0].set(1)?; 21 Ok(()) 22 } 23}

'staticにできない場合は、おとなしくStringを返した方が楽だと思います(特に初心者のうちは).

?演算子を使わなければエラーが出ない

おそらくこれは勘違いじゃないかなと.

投稿2021/09/25 16:24

編集2021/09/25 17:03
退会済みユーザー

退会済みユーザー

総合スコア0

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

tatsuya6502

2021/09/26 01:30

横から失礼します。「?演算子を使わなければエラーが出ない」件ですが、func関数の1行目を以下のようにunwrap()で書き換えることを言っているのかもしれません。 self.field[0].set(0).unwrap(); こう書いた場合は、疑似コードにあるset関数呼び出し時の'bと、func関数戻り値の'aが無関係になりますので、&mut selfの借用期間が短くなりエラーが出なくなるはずです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問