Result
Result является более богатой версией типа Option, тип который описывает возможную ошибку вместо возможного её отсутствия.
Result<T, E> имеет два возможных значения:
Ok(T): Значение типаTErr(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(()) }