Оператор Rust упрощает обработку ошибок

RustRustBeginner
Практиковаться сейчас

This tutorial is from open-source community. Access the source code

💡 Этот учебник переведен с английского с помощью ИИ. Чтобы просмотреть оригинал, вы можете перейти на английский оригинал

Введение

В этом лабораторном задании оператор ? представлен в качестве способа упрощения кода при цепочке результатов. Он используется в конце выражения, возвращающего Result, и упрощает код, автоматически обрабатывая ветви Err и Ok. Приведенный пример демонстрирует, как использовать оператор ? в Rust для обработки различных математических операций и их возможных ошибок.

Примечание: Если лабораторное задание не задает имя файла, вы можете использовать любое имя файла, которое хотите. Например, вы можете использовать main.rs, скомпилировать и запустить его с помощью rustc main.rs &&./main.


Skills Graph

%%%%{init: {'theme':'neutral'}}%%%% flowchart RL rust(("Rust")) -.-> rust/BasicConceptsGroup(["Basic Concepts"]) rust(("Rust")) -.-> rust/DataTypesGroup(["Data Types"]) rust(("Rust")) -.-> rust/FunctionsandClosuresGroup(["Functions and Closures"]) rust(("Rust")) -.-> rust/DataStructuresandEnumsGroup(["Data Structures and Enums"]) rust(("Rust")) -.-> rust/ErrorHandlingandDebuggingGroup(["Error Handling and Debugging"]) rust(("Rust")) -.-> rust/AdvancedTopicsGroup(["Advanced Topics"]) rust/BasicConceptsGroup -.-> rust/variable_declarations("Variable Declarations") rust/DataTypesGroup -.-> rust/floating_types("Floating-point Types") rust/DataTypesGroup -.-> rust/string_type("String Type") rust/FunctionsandClosuresGroup -.-> rust/function_syntax("Function Syntax") rust/FunctionsandClosuresGroup -.-> rust/expressions_statements("Expressions and Statements") rust/DataStructuresandEnumsGroup -.-> rust/method_syntax("Method Syntax") rust/ErrorHandlingandDebuggingGroup -.-> rust/panic_usage("panic! Usage") rust/AdvancedTopicsGroup -.-> rust/operator_overloading("Traits for Operator Overloading") subgraph Lab Skills rust/variable_declarations -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} rust/floating_types -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} rust/string_type -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} rust/function_syntax -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} rust/expressions_statements -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} rust/method_syntax -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} rust/panic_usage -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} rust/operator_overloading -.-> lab-150172{{"Оператор Rust упрощает обработку ошибок"}} end

?

Цепочка результатов с использованием match может стать довольно запутанной; к счастью, оператор ? можно использовать, чтобы снова сделать все красиво. ? используется в конце выражения, возвращающего Result, и эквивалентен выражению match, где ветвь Err(err) расширяется до раннего return Err(From::from(err)), а ветвь Ok(ok) расширяется до выражения ok.

mod checked {
    #[derive(Debug)]
    enum MathError {
        DivisionByZero,
        NonPositiveLogarithm,
        NegativeSquareRoot,
    }

    type MathResult = Result<f64, MathError>;

    fn div(x: f64, y: f64) -> MathResult {
        if y == 0.0 {
            Err(MathError::DivisionByZero)
        } else {
            Ok(x / y)
        }
    }

    fn sqrt(x: f64) -> MathResult {
        if x < 0.0 {
            Err(MathError::NegativeSquareRoot)
        } else {
            Ok(x.sqrt())
        }
    }

    fn ln(x: f64) -> MathResult {
        if x <= 0.0 {
            Err(MathError::NonPositiveLogarithm)
        } else {
            Ok(x.ln())
        }
    }

    // Промежуточная функция
    fn op_(x: f64, y: f64) -> MathResult {
        // Если `div` "неудается", то `DivisionByZero` будет `return`ed
        let ratio = div(x, y)?;

        // Если `ln` "неудается", то `NonPositiveLogarithm` будет `return`ed
        let ln = ln(ratio)?;

        sqrt(ln)
    }

    pub fn op(x: f64, y: f64) {
        match op_(x, y) {
            Err(why) => panic!("{}", match why {
                MathError::NonPositiveLogarithm
                    => "логарифм неположительного числа",
                MathError::DivisionByZero
                    => "деление на ноль",
                MathError::NegativeSquareRoot
                    => "квадратный корень из отрицательного числа",
            }),
            Ok(value) => println!("{}", value),
        }
    }
}

fn main() {
    checked::op(1.0, 10.0);
}

Обязательно проверьте документацию, так как есть много методов для сопоставления/композиции Result.

Резюме

Поздравляем! Вы завершили лабораторную работу по ?. Вы можете выполнить больше лабораторных работ в LabEx, чтобы улучшить свои навыки.