<style>
.textleft {
text-align:left;
}
.reveal, .reveal h1, .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 {
font-family:Arial, Microsoft JhengHei;}
.reveal .progress {
height: 14px !important;
}
.progress span {
background: url() repeat-x !important;
}
.progress span:after, .progress span.nyancat {
content: "";
background: url() ;
width: 34px !important;
height: 21px !important;
border: none !important;
float: right;
margin-top: -7px;
margin-right: -10px;
}
.my-icon-css {
width: 100%;
height: 200px;
background-position:center;
background-repeat: no-repeat;
background-image:url('https://mir-s3-cdn-cf.behance.net/project_modules/disp/fe36cc42774743.57ee5f329fae6.gif');
}
</style>
<!-- .slide: data-transition="slide" -->
# <h2>ch9. Error Handling</h2>
<div class="my-icon-css"></div>
@Kais(VagrantPi)
###### tags: `Rust`, `簡報`, `Rust 讀書會`
---
<!-- .slide: data-transition="convex" -->
## Agenda
- Unrecoverable Errors with panic!
- Recoverable Errors with Result
- To panic! or Not To panic!
---
<!-- .slide: data-transition="convex" -->
## Unrecoverable Errors with panic!
----
<!-- .slide: data-transition="convex" -->
### Rust 中的錯誤
- 可恢復錯誤(recoverable)
<!--可恢復錯誤通常代跟 user 報錯或是 retry 是合理的情况,比如上傳檔案未找到檔案-->
- 不可恢復錯誤(unrecoverable)
<!--不可恢復錯誤通常是 bug 的同義詞,比如嘗試讀取陣列範圍外的位置-->
----
<!-- .slide: data-transition="convex" -->
### Panic 又可分為兩種模式:
- 展開(unwinding)預設值
<!--Rust 會回朔並清理它遇到的每一个函式用過的資源(變數啥的)-->
- 終止(abort)
<!--不會清理數據就直接退出,這些使用的内存需要由操作系统来清理,build 出來的 binery 會比較小-->
*Cargo.toml 的 [profile] 部分增加 panic = ‘abort’*
----
<!-- .slide: data-transition="convex" -->
```javascript=
fn main() {
panic!("crash and burn");
}
```
```
$ cargo run
Compiling rust v0.1.0 (file:///Users/kais.lin/rust)
Finished dev [unoptimized + debuginfo] target(s) in 3.42 secs
Running `target/debug/rust`
thread 'main' panicked at 'crash and burn', main.rs:2:5
note: Run with `RUST_BACKTRACE=1` for a backtrace.
```
<!--不過在其他情況下,panic 會發生在呼叫的函數中,此時錯誤訊息會是指向該函式,而不是我們呼叫的那一行,這時可以使用 backtrace 來尋找-->
----
<!-- .slide: data-transition="convex" -->
### 使用 panic! 的 backtrace
```javascript=
fn main() {
let v = vec![1, 2, 3];
v[99];
}
```
```
$ cargo run
Compiling rust v0.1.0 (file:///Users/kais.lin/rust)
Finished dev [unoptimized + debuginfo] target(s) in 0.34 secs
Running `target/debug/rust`
thread 'main' panicked at 'index out of bounds: the len is 3 but the index is 99', /Users/travis/build/rust-lang/rust/src/libcore/slice/mod.rs:865:10
note: Run with `RUST_BACKTRACE=1` for a backtrace.
```
----
<!-- .slide: data-transition="convex" -->
```
$ RUST_BACKTRACE=1 cargo run
Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs
Running `target/debug/rust`
thread 'main' panicked at 'index out of bounds: the len is 3 but the index is 99', /Users/travis/build/rust-lang/rust/src/libcore/slice/mod.rs:865:10
stack backtrace:
0: std::sys::unix::backtrace::tracing::imp::unwind_backtrace
at libstd/sys/unix/backtrace/tracing/gcc_s.rs:49
1: std::sys_common::backtrace::print
at libstd/sys_common/backtrace.rs:71
at libstd/sys_common/backtrace.rs:59
2: std::panicking::default_hook::{{closure}}
at libstd/panicking.rs:207
3: std::panicking::default_hook
at libstd/panicking.rs:223
4: std::panicking::begin_panic
at libstd/panicking.rs:402
5: std::panicking::try::do_call
at libstd/panicking.rs:349
6: std::panicking::try::do_call
at libstd/panicking.rs:325
7: core::ptr::drop_in_place
at libcore/panicking.rs:72
8: core::ptr::drop_in_place
at libcore/panicking.rs:58
9: <usize as core::slice::SliceIndex<[T]>>::index
at /Users/travis/build/rust-lang/rust/src/libcore/slice/mod.rs:865
10: core::slice::<impl core::ops::index::Index<I> for [T]>::index
at /Users/travis/build/rust-lang/rust/src/libcore/slice/mod.rs:767
11: <alloc::vec::Vec<T> as core::ops::index::Index<I>>::index
at /Users/travis/build/rust-lang/rust/src/liballoc/vec.rs:1648
12: rust::main
at ./main.rs:4
13: std::rt::lang_start::{{closure}}
at /Users/travis/build/rust-lang/rust/src/libstd/rt.rs:74
14: std::panicking::try::do_call
at libstd/rt.rs:59
at libstd/panicking.rs:306
15: panic_unwind::dwarf::eh::read_encoded_pointer
at libpanic_unwind/lib.rs:102
16: <std::io::Write::write_fmt::Adaptor<'a, T> as core::fmt::Write>::write_str
at libstd/panicking.rs:285
at libstd/panic.rs:361
at libstd/rt.rs:58
17: std::rt::lang_start
at /Users/travis/build/rust-lang/rust/src/libstd/rt.rs:74
18: rust::main
```
```
12: rust::main
at ./main.rs:4
```
---
<!-- .slide: data-transition="convex" -->
## Recoverable Errors with Result
----
<!-- .slide: data-transition="convex" -->
### 面對它
使用 `Result` 類型來處理潛在錯誤
```javascript
enum Result<T, E> {
Ok(T),
Err(E),
}
```
如何知道某某函數會回傳 `Rusult` 呢?除文件之外,書中也提到了另一個方法
----
<!-- .slide: data-transition="convex" -->
### 執行它
```javascript=
use std::fs::File;
fn main() {
let f = File::open("hello.txt");
}
```
```
error[E0308]: mismatched types
--> src/main.rs:4:18
|
4 | let f: u32 = File::open("hello.txt");
| ^^^^^^^^^^^^^^^^^^^^^^^ expected u32, found enum
`std::result::Result`
|
= note: expected type `u32`
found type `std::result::Result<std::fs::File, std::io::Error>`
```
----
<!-- .slide: data-transition="convex" -->
### 處理它
```javascript=
use std::fs::File;
fn main() {
let f = File::open("hello.txt");
let f = match f {
Ok(file) => file,
Err(error) => {
panic!("There was a problem opening the file: {:?}", error)
},
};
}
```
----
<!-- .slide: data-transition="convex" -->
```javascript=
use std::fs::File;
use std::io::ErrorKind;
fn main() {
let f = File::open("hello.txt");
let f = match f {
Ok(file) => file,
Err(ref error) if error.kind() == ErrorKind::NotFound => {
match File::create("hello.txt") {
Ok(fc) => fc,
Err(e) => {
panic!(
"Tried to create file but there was a problem: {:?}",
e
)
},
}
},
Err(error) => {
panic!(
"There was a problem opening the file: {:?}",
error
)
},
};
}
```
if error.kind() == ErrorKind::NotFound 被稱為 `match guard`,用來更完善 match 的而外條件判斷
----
<!-- .slide: data-transition="convex" -->
### 失敗時 panic 的簡寫:unwrap
```javascript=
use std::fs::File;
fn main() {
let f = File::open("hello.txt").unwrap();
}
```
```
thread 'main' panicked at 'called `Result::unwrap()` on an `Err` value: Error {
repr: Os { code: 2, message: "No such file or directory" } }',
src/libcore/result.rs:906:4
```
----
<!-- .slide: data-transition="convex" -->
### 失敗時 panic 的簡寫:expect
```javascript=
use std::fs::File;
fn main() {
let f = File::open("hello.txt").expect("Failed to open hello.txt");
}
```
```
thread 'main' panicked at 'Failed to open hello.txt: Error { repr: Os { code:
2, message: "No such file or directory" } }', src/libcore/result.rs:906:4
```
----
<!-- .slide: data-transition="convex" -->
### 傳播(propagating)錯誤
簡單說就是 function 會回傳 error,這樣可以讓呼叫他的地方能夠判斷要不要做錯誤處裡
```javascript=
use std::io;
use std::io::Read;
use std::fs::File;
fn read_username_from_file() -> Result<String, io::Error> {
let f = File::open("hello.txt");
let mut f = match f {
Ok(file) => file,
Err(e) => return Err(e),
};
let mut s = String::new();
match f.read_to_string(&mut s) {
Ok(_) => Ok(s),
Err(e) => Err(e),
}
}
```
----
<!-- .slide: data-transition="convex" -->
### 傳播錯誤的簡寫: ?
```javascript=
use std::io;
use std::io::Read;
use std::fs::File;
fn read_username_from_file() -> Result<String, io::Error> {
let mut f = File::open("hello.txt")?;
let mut s = String::new();
f.read_to_string(&mut s)?;
Ok(s)
}
```
----
<!-- .slide: data-transition="convex" -->
### 與 match 的差別
原文理解不能,以下是我個人理解,請大大們勘誤
`?` 會將錯誤值丟給 from 函數(定義於 From trait 中),可以將 Result 回傳的錯誤型別,轉換成該 fn 回傳的錯誤型別
----
<!-- .slide: data-transition="convex" -->
```javascript=
use std::io::{self, Read};
use std::num;
enum CliError {
IoError(io::Error),
ParseError(num::ParseIntError),
}
impl From<io::Error> for CliError {
fn from(error: io::Error) -> Self {
CliError::IoError(error)
}
}
impl From<num::ParseIntError> for CliError {
fn from(error: num::ParseIntError) -> Self {
CliError::ParseError(error)
}
}
fn open_and_parse_file(file_name: &str) -> Result<i32, CliError> {
let mut file = std::fs::File::open("test")?;
let mut contents = String::new();
file.read_to_string(&mut contents)?;
let num: i32 = contents.trim().parse()?;
Ok(num)
}
```
> [trait.From](https://doc.rust-lang.org/std/convert/trait.From.html)
----
<!-- .slide: data-transition="convex" -->
### ? 只能被用于返回 Result 的函数
```javascript=
use std::fs::File;
fn main() {
let f = File::open("hello.txt")?;
}
```
```
error[E0277]: the trait bound `(): std::ops::Try` is not satisfied
--> src/main.rs:4:13
|
4 | let f = File::open("hello.txt")?;
| ------------------------
| |
| the `?` operator can only be used in a function that returns
`Result` (or another type that implements `std::ops::Try`)
| in this macro invocation
|
= help: the trait `std::ops::Try` is not implemented for `()`
= note: required by `std::ops::Try::from_error`
```
---
<!-- .slide: data-transition="convex" -->
## To panic! or Not To panic!
----
<!-- .slide: data-transition="convex" -->
### Examples, Prototype Code, and Tests
有時候只是要快速的驗證或者是 demo,可以先 unwrap 之類的方法,未來再來實現這部廢的錯誤處理
----
<!-- .slide: data-transition="convex" -->
### 當你比編譯器知道更多情況
```javascript=
use std::net::IpAddr;
let home: IpAddr = "127.0.0.1".parse().unwrap();
```
即便是寫死的,而且確定是可分析的 ip,編譯器仍會覺得還是可能會出現 Err 因此要我們處理 Result,這邊種情況下使用 unwrap 是合適的
----
<!-- .slide: data-transition="convex" -->
### 當輸入值錯誤時
預期外的傳入值,可能導致存取非當前資料的記憶體空間,而出現安全性問題
不過透過 Rust 的 type system 就會為你做很多檢查了(EX: u32 接到的值不會有負的)
<!--程式在對值進行操作時,應該先驗證其為有效值,否則就 panic!,主要是因為安全原因,嘗試操作無效數據可能會暴露程式漏洞,嘗試訪問不屬於當前數據的記憶體空間是一個常見的隱患-->
----
<!-- .slide: data-transition="convex" -->
### 建立自訂義的型別驗證
v0.0.1
```javascript=
loop {
// --snip--
let guess: i32 = match guess.trim().parse() {
Ok(num) => num,
Err(_) => continue,
};
if guess < 1 || guess > 100 {
println!("The secret number will be between 1 and 100.");
continue;
}
match guess.cmp(&secret_number) {
// --snip--
}
```
----
<!-- .slide: data-transition="convex" -->
不過這樣但這樣每次都要進一次判斷式,如果很多的函式都需要這樣的檢查,這樣寫就恨很冗餘(也許也會小小的引響到效能)
----
<!-- .slide: data-transition="convex" -->
```javascript=
pub struct Guess {
value: u32,
}
impl Guess {
pub fn new(value: u32) -> Guess {
if value < 1 || value > 100 {
panic!("Guess value must be between 1 and 100, got {}.", value);
}
Guess {
value
}
}
pub fn value(&self) -> u32 {
self.value
}
}
```
----
<!-- .slide: data-transition="convex" -->
v0.0.2
```diff=
extern crate rand;
use std::io;
use std::cmp::Ordering;
use rand::Rng;
fn main() {
println!("Guess the number!");
let secret_number = rand::thread_rng().gen_range(1, 101);
loop {
println!("Please input your guess.");
let mut guess = String::new();
io::stdin().read_line(&mut guess)
.expect("Failed to read line");
let guess: u32 = match guess.trim().parse() {
- Ok(num) => num,
+ Ok(num) => Guess::new(num).value(),
Err(_) => continue,
};
println!("You guessed: {}", guess);
match guess.cmp(&secret_number) {
Ordering::Less => println!("Too small!"),
Ordering::Greater => println!("Too big!"),
Ordering::Equal => {
println!("You win!");
break;
}
}
}
}
+ pub struct Guess {
+ value: u32,
+ }
+
+ impl Guess {
+ pub fn new(value: u32) -> Guess {
+ if value < 1 || value > 100 {
+ panic!("Guess value must be between 1 and +100, got {}.", value);
+ }
+
+ Guess {
+ value
+ }
+ }
+
+ pub fn value(&self) -> u32 {
+ self.value
+ }
+ }
```
---
<!-- .slide: data-transition="convex" -->
## Thanks for listening ~
{"metaMigratedAt":"2023-06-14T18:22:36.042Z","metaMigratedFrom":"Content","title":"<h2>ch9. Error Handling</h2>","breaks":true,"contributors":"[{\"id\":\"69ade472-3ed3-499d-8a69-767243a31621\",\"add\":26527,\"del\":7855}]"}