Result
Result
является более богатой версией типа Option
, тип который описывает возможную ошибку вместо возможного её отсутствия.
Result<T, E>
имеет два возможных значения:
Ok(T)
: Значение типаT
Err(E)
: Ошибка обработки элемента, типаE
По соглашению, ожидаемый результат Ok
, тогда как не ожидаемый - Err
.
Подобно Option
, Result
имеет множество ассоциированных с ним методов. Например, unwrap()
или возвращает T
, или вызывает panic
.
Для обработки результата у Result
существует множество комбинаторов, которые совпадают с комбинаторами Option
.
При работе с Rust вы, скорее всего, столкнётесь с методами, которые возвращают тип Result
, например метод parse()
. Не всегда
можно разобрать строку в другой тип, поэтому parse()
возвращает Result
, указывающий на возможный сбой.
Давайте посмотрим, что происходит, когда мы успешно и безуспешно попытаемся преобразовать строку с помощью parse()
:
fn multiply(first_number_str: &str, second_number_str: &str) -> i32 { // Давайте попробуем использовать `unwrap()` чтобы получить число. Он нас укусит? let first_number = first_number_str.parse::<i32>().unwrap(); let second_number = second_number_str.parse::<i32>().unwrap(); first_number * second_number } fn main() { let twenty = multiply("10", "2"); println!("удовоенное {}", twenty); let tt = multiply("t", "2"); println!("удвоенное {}", tt); }
При неудаче, parse()
оставляет на с ошибкой, с
которой unwrap()
вызывает panic
. Дополнительно, panic
завершает нашу программу и
предоставляет неприятное сообщение об ошибке.
Для повышения качества наших сообщений об ошибка, мы должны более явно указать возвращаемый тип и рассмотреть возможной явной обработки ошибок.
Использование Result
в main
Также Result
может быть возвращаемым типом функции main
, если это указано явно. Обычно функция main
имеют следующую форму:
fn main() { println!("Hello World!"); }
Однако main
также может и возвращать тип Result
. Если ошибка происходит в пределах функции main
, то она возвращает код ошибки и выводит отладочное представление ошибки (используя типаж Debug
). Следующий пример показывает такой сценарий и затрагивает аспекты, описанные в последующем разделе.
use std::num::ParseIntError; fn main() -> Result<(), ParseIntError> { let number_str = "10"; let number = match number_str.parse::<i32>() { Ok(number) => number, Err(e) => return Err(e), }; println!("{}", number); Ok(()) }