色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Rust中的錯誤處理方法

jf_wN0SrCdH ? 來源:Rust語言中文社區 ? 2023-02-20 09:37 ? 次閱讀

Rust中的錯誤處理

Result枚舉

Rust 中沒有提供類似于 Java、C++ 中的 Exception 機制,而是使用Result枚舉的方式來實現:


	

pub enum Result { /// Contains the success value Ok(T), /// Contains the error value Err(E), }

在使用時:

  • 如果無錯誤則使用Ok(T)返回;

  • 如果存在錯誤,則使用Err(E)包裝錯誤類型返回;

例如:

examples/0_result.rs


	

#[derive(Debug)] pub enum MyError { Internal(String), InvalidId(String), } fn add(num: i64) -> Result<i64, MyError> { if num < 0 { Err(MyError::InvalidId(String::from("Invalid num!"))) } else { Ok(num + 100000) } } fn main() -> Result<(), MyError> { // fetch_id(-1)?; let res = add(1)?; println!("{}", res); Ok(()) }

上面的代碼首先通過 MyError 枚舉定義了多個可能會出現的錯誤;

隨后,在add函數中:

  • 當 num 小于 0 時返回錯誤;

  • 否則給 num 增加 100000 并返回;

在上面的let res = add(1)?;中使用了?操作符,他相當于是一個語法糖:

  • 如果被調函數正常返回則調用unwrap獲取其值;

  • 反之,則將被調函數的錯誤直接向上返回(相當于直接 return Err);

即上面的語法糖相當于:


	

let res = match add() { Ok(id) => id, Err(err) => { return Err(err); } };

錯誤類型轉換

上面簡單展示了 Rust 中錯誤的使用;

由于 Rust 是強類型的語言,因此如果在一個函數中使用?返回了多個錯誤,并且他們的類型是不同的,還需要對返回的錯誤類型進行轉換,轉為相同的類型!

例如下面的例子:


	

#[derive(Debug)] pub enum MyError { ReadError(String), ParseError(String), } fn read_file() -> Result<i64, MyError> { // Error: Could not get compiled! let content = fs::read_to_string("/tmp/id")?; let id = content.parse::<i64>()?; } fn main() -> Result<(), MyError> { let id = read_file()?; println!("id: {}", id); Ok(()) }

上面的例子無法編譯通過,原因在于:read_to_stringparse返回的是不同類型的錯誤!

因此,如果要能返回,我們需要對每一個錯誤進行轉換,轉為我們所定義的 Error 類型;

例如:

examples/1_error_convert.rs


	

fn read_file() -> Result<i64, MyError> { // Error: Could not get compiled! // let content = fs::read_to_string("/tmp/id")?; // let id = content.parse::()?; // Method 1: Handling error explicitly! let content = match std::read_to_string("/tmp/id") { Ok(content) => content, Err(err) => { return Err(MyError::ReadError(format!("read /tmp/id failed: {}", err))); } }; let content = content.trim(); println!("read content: {}", content); // Method 2: Use map_err to transform error type let id = content .parse::<i64>() .map_err(|err| MyError::ParseError(format!("parse error: {}", err)))?; Ok(id) }

上面展示了兩種不同的轉換 Error 的方法:

方法一通過 match 匹配手動的對read_to_string函數的返回值進行處理,如果發生了 Error,則將錯誤轉為我們指定類型的錯誤;

方法二通過map_err的方式,如果返回的是錯誤,則將其轉為我們指定的類型,這時就可以使用?返回了;

相比之下,使用 map_err 的方式,代碼會清爽很多!

From Trait

上面處理錯誤的方法,每次都要對錯誤的類型進行轉換,比較麻煩;

Rust 中提供了 From Trait,在進行類型匹配時,如果提供了從一個類型轉換為另一個類型的方法(實現了某個類型的 From Trait),則在編譯階段,編譯器會調用響應的函數,直接將其轉為相應的類型!

例如:

examples/2_from_trait.rs


	

#[derive(Debug)] pub enum MyError { ReadError(String), ParseError(String), } impl From for MyError { fn from(source: std::Error) -> Self { MyError::ReadError(source.to_string()) } } impl From for MyError { fn from(source: std::ParseIntError) -> Self { MyError::ParseError(source.to_string()) } } fn read_file() -> Result<i64, MyError> { let _content = fs::read_to_string("/tmp/id")?; let content = _content.trim(); let id = content.parse::<i64>()?; Ok(id) } fn main() -> Result<(), MyError> { let id = read_file()?; println!("id: {}", id); Ok(()) }

在上面的代碼中,我們為 MyError 類型的錯誤分別實現了轉換為std::Errorstd::ParseIntError類型的 From Trait;

因此,在 read_file 函數中就可以直接使用?向上返回錯誤了!

但是上面的方法需要為每個錯誤實現 From Trait 還是有些麻煩,因此出現了thiserror以及anyhow庫來解決這些問題;

其他第三方庫

thiserror

上面提到了我們可以為每個錯誤實現 From Trait 來直接轉換錯誤類型,thiserror庫就是使用這個邏輯;

我們可以使用 thiserror 庫提供的宏來幫助我們生成到對應類型的 Trait;

例如:

examples/3_thiserror.rs


	

#[derive(thiserror::Error, Debug)] pub enum MyError { #[error("io error.")] IoError(#[from] std::Error), #[error("parse error.")] ParseError(#[from] std::ParseIntError), } fn read_file() -> Result<i64, MyError> { // Could get compiled! let content = fs::read_to_string("/tmp/id")?; let id = content.parse::<i64>()?; Ok(id) } fn main() -> Result<(), MyError> { let id = read_file()?; println!("id: {}", id); Ok(()) }

我們只需要對我們定義的類型進行宏標注,在編譯時這些宏會自動展開并實現對應的 Trait;

展開后的代碼如下:


	

#![feature(prelude_import)] #[prelude_import] use std::*; #[macro_use] extern crate std; use std::fs; pub enum MyError { #[error("io error.")] IoError(#[from] std::Error), #[error("parse error.")] ParseError(#[from] std::ParseIntError), } #[allow(unused_qualifications)] impl std::Error for MyError { fn source(&self) -> std::Option<&(dyn std::Error + 'static)> { use thiserror::AsDynError; #[allow(deprecated)] match self { MyError::IoError { 0: source, .. } => std::Option::Some(source.as_dyn_error()), MyError::ParseError { 0: source, .. } => { std::Option::Some(source.as_dyn_error()) } } } } #[allow(unused_qualifications)] impl std::Display for MyError { fn fmt(&self, __formatter: &mut std::Formatter) -> std::Result { #[allow(unused_variables, deprecated, clippy::used_underscore_binding)] match self { MyError::IoError(_0) => { let result = __formatter.write_fmt(::new_v1(&["io error."], &[])); result } MyError::ParseError(_0) => { let result = __formatter.write_fmt(::new_v1(&["parse error."], &[])); result } } } } #[allow(unused_qualifications)] impl std::From for MyError { #[allow(deprecated)] fn from(source: std::Error) -> Self { MyError::IoError { 0: source } } } #[allow(unused_qualifications)] impl std::From for MyError { #[allow(deprecated)] fn from(source: std::ParseIntError) -> Self { MyError::ParseError { 0: source } } } #[automatically_derived] #[allow(unused_qualifications)] impl ::Debug for MyError { fn fmt(&self, f: &mut ::Formatter) -> ::Result { match (&*self,) { (&MyError::IoError(ref __self_0),) => { ::debug_tuple_field1_finish(f, "IoError", &&*__self_0) } (&MyError::ParseError(ref __self_0),) => { ::debug_tuple_field1_finish(f, "ParseError", &&*__self_0) } } } } fn read_file() -> Result<i64, MyError> { let content = fs::read_to_string("/tmp/id")?; let id = content.parse::<i64>()?; Ok(id) } #[allow(dead_code)] fn main() -> Result<(), MyError> { let id = read_file()?; { ::new_v1( &["id: ", " "], &[::new_display(&id)], )); }; Ok(()) } #[rustc_main] pub fn main() -> () { extern crate test; test::test_main_static(&[]) }

可以看到實際上就是為 MyError 實現了對應錯誤類型的 From Trait;

thiserror 庫的這種實現方式,還需要為類型指定要轉換的錯誤類型;

而下面看到的 anyhow 庫,可以將錯誤類型統一為同一種形式;

anyhow

如果你對 Go 中的錯誤類型不陌生,那么你就可以直接上手 anyhow 了!

來看下面的例子:

examples/4_anyhow.rs


	

use anyhow::Result; use std::fs; fn read_file() -> Result<i64> { // Could get compiled! let content = fs::read_to_string("/tmp/id")?; let id = content.parse::<i64>()?; Ok(id) } fn main() -> Result<()> { let id = read_file()?; println!("id: {}", id); Ok(()) }

注意到,上面的 Result 類型為anyhow::Result,而非標準庫中的 Result 類型!

anyhowResult實現了ContextTrait:


	

impl Context for Result where E: ext::StdError + Send + Sync + 'static, { fn context(self, context: C) -> Result where C: Display + Send + Sync + 'static, { // Not using map_err to save 2 useless frames off the captured backtrace // in ext_context. match self { Ok(ok) => Ok(ok), Err(error) => Err(error.ext_context(context)), } } fn with_context(self, context: F) -> Result where C: Display + Send + Sync + 'static, F: FnOnce() -> C, { match self { Ok(ok) => Ok(ok), Err(error) => Err(error.ext_context(context())), } } }

Context中提供了context函數,并且將原來的Result轉成了Result;

因此,最終將錯誤類型統一為了anyhow::Error類型;

附錄

源代碼:

  • https://github.com/JasonkayZK/rust-learn/tree/error

審核編輯:湯梓紅


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • JAVA
    +關注

    關注

    19

    文章

    2974

    瀏覽量

    104983
  • 函數
    +關注

    關注

    3

    文章

    4345

    瀏覽量

    62882
  • C++
    C++
    +關注

    關注

    22

    文章

    2114

    瀏覽量

    73793
  • 枚舉
    +關注

    關注

    0

    文章

    16

    瀏覽量

    4613
  • Rust
    +關注

    關注

    1

    文章

    230

    瀏覽量

    6641

原文標題:[Rust筆記] Rust中的錯誤處理

文章出處:【微信號:Rust語言中文社區,微信公眾號:Rust語言中文社區】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    嵌入式編程錯誤處理機制設計

    本文主要總結嵌入式系統C語言編程,主要的錯誤處理方式。文中涉及的代碼運行環境如下。
    發表于 04-28 09:59 ?787次閱讀
    嵌入式編程<b class='flag-5'>錯誤處理</b>機制設計

    嵌入式系統C語言編程主要的錯誤處理方式

    本文主要總結嵌入式系統C語言編程,主要的錯誤處理方式。
    發表于 07-24 16:40 ?984次閱讀
    嵌入式系統C語言編程<b class='flag-5'>中</b>主要的<b class='flag-5'>錯誤處理</b>方式

    Rust語言中錯誤處理的機制

    可能的錯誤,實際運行仍然可能出現各種各樣的錯誤,比如文件不存在、網絡連接失敗等等。對于這些不可預測的錯誤,我們必須使用錯誤處理機制來進行
    的頭像 發表于 09-19 14:54 ?1472次閱讀

    嵌入式C編程常用的異常錯誤處理

    嵌入式C編程,異常錯誤處理是確保系統穩定性和可靠性的重要部分。以下是一些常見的異常錯誤處理方法及其詳細說明和示例: 1. 斷言 (Assertions) 斷言用于在開發階段捕獲程
    發表于 08-06 14:32

    labviEW錯誤處理的問題

    為什么這個程序在啟用自動錯誤處理和C:\data.txt不存在的情況下,沒有顯示錯誤對話框???
    發表于 04-01 10:03

    AF錯誤處理

    想問一下關于AF的錯誤處理,例如我進行串口通訊,打開串口錯誤,但是我不想停止AF,想繼續嘗試連接要怎么做?
    發表于 02-03 15:44

    LabVIEW錯誤處理

    如何合理使用 LabVIEW 的自定義錯誤處理功能;對于可預見的錯誤,是否可以選擇直 接忽略,或者前幾次嘗試忽略直到該特定錯誤出現很多次后才通知用戶需要糾正該
    發表于 05-24 11:07 ?6次下載

    Spring Boot框架錯誤處理

    》 《strong》翻譯《/strong》:雁驚寒《/p》 《/blockquote》《p》《em》摘要:本文通過實例介紹了使用Spring Boot在設計API的時候如何正確地對異常進行處理。以下是譯文《/em》《/p》《p》API在提供錯誤消息的同時進行適當的
    發表于 09-28 15:31 ?0次下載

    嵌入式系統C語言編程錯誤處理資料總結

    本文主要總結嵌入式系統C語言編程,主要的錯誤處理方式。文中涉及的代碼運行環境如下:
    發表于 11-28 10:39 ?1952次閱讀

    Rust代碼啟發之返回值異常錯誤處理

    這樣的代碼,錯誤處理代碼和業務邏輯交織在一起,也容易忽略處理錯誤。以及把返回值只用于錯誤返回,有點浪費的感覺。因為很多時候把計算結果作為返回值,更符合思考的邏輯。
    的頭像 發表于 09-22 09:24 ?2203次閱讀
    <b class='flag-5'>Rust</b>代碼啟發之返回值異常<b class='flag-5'>錯誤處理</b>

    rust語言基礎學習: rust錯誤處理

    錯誤是軟件不可避免的,所以 Rust 有一些處理出錯情況的特性。在許多情況下,Rust 要求你承認錯誤
    的頭像 發表于 05-22 16:28 ?2178次閱讀

    RS232通信時怎么處理錯誤?RS232通信中的錯誤處理方法

    RS232通信時怎么處理錯誤?RS232通信中的錯誤處理方法? RS232通信是一種電氣標準,它定義了計算機和串行通信設備之間的通信協議。盡管RS232通信很穩定,但仍然可能會出現
    的頭像 發表于 10-17 16:33 ?3109次閱讀

    西門子博圖:錯誤處理機制概覽

    可通過以下幾種不同的錯誤處理機制進行參數跟蹤或編程或訪問錯誤
    的頭像 發表于 11-25 11:35 ?3019次閱讀
    西門子博圖:<b class='flag-5'>錯誤處理</b>機制概覽

    C語言中的錯誤處理機制解析

    C 語言不提供對錯誤處理的直接支持,但是作為一種系統編程語言,它以返回值的形式允許您訪問底層數據。
    的頭像 發表于 02-26 11:19 ?546次閱讀

    socket編程錯誤處理技巧

    Socket編程是網絡編程的基礎,它允許程序之間通過TCP/IP協議進行通信。然而,網絡通信是不穩定的,可能會遇到各種問題,如網絡延遲、連接中斷、數據丟失等。 錯誤處理的重要性 提高程序的健壯性
    的頭像 發表于 11-01 17:47 ?918次閱讀
    主站蜘蛛池模板: 18女下面流水不遮网站免费 | 亚洲 欧美 中文 日韩 视频 | 果冻传媒剧情在线观看 | 国内精品久久久久影院网站 | 最新无码国产在线视频9299 | 毛片在线全部免费观看 | 韩国女主播内部vip自带氏巾 | 泡妞高手在都市完整版视频免费 | 日本粉嫩学生毛绒绒 | 中文字幕在线观看亚洲视频 | 孕妇bbwbbwbbwbbw超清 | 青柠在线视频 | 国产精品xxxav免费视频 | 久草在线精彩免费视频 | 午夜无码国产理论在线 | 717影院理论午夜伦不卡久久 | 9277在线观看免费高清完整版 | 久久视热频国产这里只有精品23 | 男男女女爽爽爽视频免费 | 亚洲AV精品乱码专区 | 精品国产高清自在线看 | 国产福利一区二区精品 | 老鸭窝毛片 | 2020最新无码国产在线视频 | 蜜柚在线观看免费高清官网视频 | 交换:年轻夫妇-HD中文字幕 | 乡村教师电影版 | 久久天天婷婷五月俺也去 | 亚洲精品国偷拍自产在线观看蜜臀 | 中国人泡妞xxxxxxxx19 | 青青草原网址 | 超碰97人人做人人爱亚洲尤物 | 欧美亚洲韩日午夜 | 国产三级在线免费 | 国产成人在线免费 | 亚洲视频第二页 | 99国产亚洲精品无码成人 | 久久精品动漫网一区二区 | 色综合a在线 | 九九热在线视频精品店 | 伊人成综合网伊人222 |