轉換錯誤型別
use std::error::Error; use std::fmt::{self, Display, Formatter}; use std::fs::{self, File}; use std::io::{self, Read}; #[derive(Debug)] enum ReadUsernameError { IoError(io::Error), EmptyUsername(String), } impl Error for ReadUsernameError {} impl Display for ReadUsernameError { fn fmt(&self, f: &mut Formatter) -> fmt::Result { match self { Self::IoError(e) => write!(f, "IO error: {e}"), Self::EmptyUsername(filename) => write!(f, "Found no username in {filename}"), } } } impl From<io::Error> for ReadUsernameError { fn from(err: io::Error) -> ReadUsernameError { ReadUsernameError::IoError(err) } } fn read_username(path: &str) -> Result<String, ReadUsernameError> { let mut username = String::with_capacity(100); File::open(path)?.read_to_string(&mut username)?; if username.is_empty() { return Err(ReadUsernameError::EmptyUsername(String::from(path))); } Ok(username) } fn main() { //fs::write("config.dat", "").unwrap(); let username = read_username("config.dat"); println!("username or error: {username:?}"); }
重要須知:
username
變數可以是Ok(string)
或Err(error)
。- 請使用
fs::write
呼叫來測試以下不同情況:沒有檔案、空白檔案、含使用者名稱的檔案。
對所有不需要是 no_std
的錯誤型別來說,實作 std::error::Error
是很好的做法,std::error::Error
會需要 Debug
和 Display
。core
的 Error
Crate 僅於 nightly 提供,因此與 no_std
尚未完全相容。
對這種錯誤型別來說,在可能的情況下實作 Clone
和 Eq
通常也很有用,不僅有利於程式庫的測試,使用者也會更輕鬆。但在本例中,我們無法輕易這麼做,因為 io::Error
並未實作 Clone
和 Eq
。