1
0
mirror of https://github.com/google/comprehensive-rust.git synced 2025-06-04 00:29:11 +02:00

61 lines
1.8 KiB
Markdown
Raw Normal View History

2022-12-21 16:36:30 +01:00
# Propagating Errors with `?`
The try-operator `?` is used to return errors to the caller. It lets you turn
the common
```rust,ignore
match some_expression {
Ok(value) => value,
Err(err) => return Err(err),
}
```
into the much simpler
```rust,ignore
some_expression?
```
2023-07-06 16:25:37 +02:00
We can use this to simplify our error handling code:
2022-12-21 16:36:30 +01:00
```rust,editable
use std::{fs, io};
use std::io::Read;
2022-12-21 16:36:30 +01:00
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),
2022-12-21 16:36:30 +01:00
};
let mut username = String::new();
match username_file.read_to_string(&mut username) {
Ok(_) => Ok(username),
Err(err) => Err(err),
2022-12-21 16:36:30 +01:00
}
}
fn main() {
//fs::write("config.dat", "alice").unwrap();
let username = read_username("config.dat");
println!("username or error: {username:?}");
2022-12-21 16:36:30 +01:00
}
```
2023-01-11 18:07:14 -08:00
<details>
Key points:
* The `username` variable can be either `Ok(string)` or `Err(error)`.
* Use the `fs::write` call to test out the different scenarios: no file, empty file, file with username.
* The return type of the function has to be compatible with the nested functions it calls. For instance,
a function returning a `Result<T, Err>` can only apply the `?` operator on a function returning a
`Result<AnyT, Err>`. It cannot apply the `?` operator on a function returning an `Option<AnyT>` or `Result<T, OtherErr>`
unless `OtherErr` implements `From<Err>`. Reciprocally, a function returning an `Option<T>` can only apply the `?` operator
on a function returning an `Option<AnyT>`.
* You can convert incompatible types into one another with the different `Option` and `Result` methods
such as `Option::ok_or`, `Result::ok`, `Result::err`.
2023-01-11 18:07:14 -08:00
</details>