使用 ? 傳播錯誤

try 運算子 ? 用於將錯誤傳回呼叫端,讓您將下列常見的程式碼:

match some_expression {
    Ok(value) => value,
    Err(err) => return Err(err),
}

轉換成以下較簡潔的程式碼:

some_expression?

We can use this to simplify our error handling code:

use std::{fs, io};
use std::io::Read;

fn read_username(path: &str) -> Result<String, io::Error> {
    let username_file_result = fs::File::open(path);
    let mut username_file = match username_file_result {
        Ok(file) => file,
        Err(err) => return Err(err),
    };

    let mut username = String::new();
    match username_file.read_to_string(&mut username) {
        Ok(_) => Ok(username),
        Err(err) => Err(err),
    }
}

fn main() {
    //fs::write("config.dat", "alice").unwrap();
    let username = read_username("config.dat");
    println!("username or error: {username:?}");
}

重要須知:

  • username 變數可以是 Ok(string)Err(error)
  • 請使用 fs::write 呼叫來測試以下不同情況:沒有檔案、空白檔案、含使用者名稱的檔案。
  • 函式的傳回型別必須與它呼叫的巢狀函式相容。舉例來說,傳回 Result<T, Err> 的函式只能在傳回 Result<AnyT, Err> 的函式上套用 ? 運算子;無法在傳回 Option<AnyT>Result<T, OtherErr> 的函式上套用該運算子,除非 OtherErr 實作了 From<Err> 則例外。反之,傳回 Option<T> 的函式只能在傳回 Option<AnyT> 的函式上套用 ? 運算子。
    • 您可以使用不同的 OptionResult 方法,例如 Option::ok_orResult::okResult::err 等,將不相容的型別轉換為其他型別。