Zoeken…


Invoering

Rust gebruikt Result<T, E> -waarden om herstelbare fouten aan te geven tijdens de uitvoering. Onherstelbare fouten veroorzaken Paniek , een onderwerp op zich.

Opmerkingen

Details van foutafhandeling worden beschreven in The Rust Programming Language (ook bekend als The Book)

Gemeenschappelijke resultaatmethoden

use std::io::{Read, Result as IoResult};
use std::fs::File;

struct Config(u8);

fn read_config() -> IoResult<String> {
    let mut s = String::new();
    let mut file = File::open(&get_local_config_path())
        // or_else closure is invoked if Result is Err.
        .or_else(|_| File::open(&get_global_config_path()))?;
    // Note: In `or_else`, the closure should return a Result with a matching
    //       Ok type, whereas in `and_then`, the returned Result should have a
    //       matching Err type.
    let _ = file.read_to_string(&mut s)?;
    Ok(s)
}

struct ParseError;

fn parse_config(conf_str: String) -> Result<Config, ParseError> {
    // Parse the config string...
    if conf_str.starts_with("bananas") {
        Err(ParseError)
    } else {
        Ok(Config(42))
    }
}

fn run() -> Result<(), String> {
    // Note: The error type of this function is String. We use map_err below to
    //       make the error values into String type
    let conf_str = read_config()
        .map_err(|e| format!("Failed to read config file: {}", e))?;
    // Note: Instead of using `?` above, we can use `and_then` to wrap the let
    //       expression below.
    let conf_val = parse_config(conf_str)
        .map(|Config(v)| v / 2) // map can be used to map just the Ok value
        .map_err(|_| "Failed to parse the config string!".to_string())?;

    // Run...

    Ok(())
}

fn main() {
    match run() {
        Ok(_) => println!("Bye!"),
        Err(e) => println!("Error: {}", e),
    }
}

fn get_local_config_path() -> String {
    let user_config_prefix = "/home/user/.config";
    // code to get the user config directory
    format!("{}/my_app.rc", user_config_prefix)
}

fn get_global_config_path() -> String {
    let global_config_prefix = "/etc";
    // code to get the global config directory
    format!("{}/my_app.rc", global_config_prefix)
}

Als de configuratiebestanden niet bestaan, voert dit uit:

Error: Failed to read config file: No such file or directory (os error 2)

Als het parseren is mislukt, geeft dit het volgende weer:

Error: Failed to parse the config string!

Opmerking: naarmate het project groeit, wordt het omslachtig om fouten met deze basismethoden ( docs ) af te handelen zonder informatie te verliezen over de oorsprong en het propagatiepad van fouten. Het is ook absoluut een slechte gewoonte om fouten vroegtijdig in tekenreeksen te converteren om meerdere fouttypen te verwerken, zoals hierboven weergegeven. Een veel betere manier is om de kist te gebruiken error-chain .

Aangepaste fouttypen

use std::error::Error;
use std::fmt;
use std::convert::From;
use std::io::Error as IoError;
use std::str::Utf8Error;

#[derive(Debug)] // Allow the use of "{:?}" format specifier
enum CustomError {
    Io(IoError),
    Utf8(Utf8Error),
    Other,
}

// Allow the use of "{}" format specifier
impl fmt::Display for CustomError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match *self {
            CustomError::Io(ref cause) => write!(f, "I/O Error: {}", cause),
            CustomError::Utf8(ref cause) => write!(f, "UTF-8 Error: {}", cause),
            CustomError::Other => write!(f, "Unknown error!"),
        }
    }
}

// Allow this type to be treated like an error
impl Error for CustomError {
    fn description(&self) -> &str {
        match *self {
            CustomError::Io(ref cause) => cause.description(),
            CustomError::Utf8(ref cause) => cause.description(),
            CustomError::Other => "Unknown error!",
        }
    }

    fn cause(&self) -> Option<&Error> {
        match *self {
            CustomError::Io(ref cause) => Some(cause),
            CustomError::Utf8(ref cause) => Some(cause),
            CustomError::Other => None,
        }
    }
}

// Support converting system errors into our custom error.
// This trait is used in `try!`.
impl From<IoError> for CustomError {
    fn from(cause: IoError) -> CustomError {
        CustomError::Io(cause)
    }
}
impl From<Utf8Error> for CustomError {
    fn from(cause: Utf8Error) -> CustomError {
        CustomError::Utf8(cause)
    }
}

Door oorzaken heenkomen

Het is vaak handig voor het opsporen van fouten om de oorzaak van een fout te achterhalen. Om een foutwaarde te onderzoeken die std::error::Error implementeert:

use std::error::Error;

let orig_error = call_returning_error();

// Use an Option<&Error>. This is the return type of Error.cause().
let mut err = Some(&orig_error as &Error);

// Print each error's cause until the cause is None.
while let Some(e) = err {
    println!("{}", e);
    err = e.cause();
}

Basisfoutrapportage en -behandeling

Result<T, E> is een enum dat twee varianten heeft: Ok(T) geeft een succesvolle uitvoering aan met een betekenisvol resultaat van type T en Err(E) geeft een optreden van een onverwachte fout aan tijdens de uitvoering, beschreven door een waarde van type E .

enum DateError {
    InvalidDay,
    InvalidMonth,
}

struct Date {
    day: u8,
    month: u8,
    year: i16,
}

fn validate(date: &Date) -> Result<(), DateError> {
    if date.month < 1 || date.month > 12 {
        Err(DateError::InvalidMonth)
    } else if date.day < 1 || date.day > 31 {
        Err(DateError::InvalidDay)
    } else {
        Ok(())
    }
}

fn add_days(date: Date, days: i32) -> Result<Date, DateError> {
    validate(&date)?; // notice `?` -- returns early on error
    // the date logic ...
    Ok(date)
}

Zie ook documenten voor meer informatie over ? operator.

Standaardbibliotheek bevat een eigenschap Error die door alle fouttypen wordt aanbevolen om te implementeren. Een voorbeeldimplementatie wordt hieronder gegeven.

use std::error::Error;
use std::fmt;

#[derive(Debug)]
enum DateError {
    InvalidDay(u8),
    InvalidMonth(u8),
}

impl fmt::Display for DateError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match self {
            &DateError::InvalidDay(day) => write!(f, "Day {} is outside range!", day),
            &DateError::InvalidMonth(month) => write!(f, "Month {} is outside range!", month),
        }
    }
}

impl Error for DateError {
    fn description(&self) -> &str {
        match self {
            &DateError::InvalidDay(_) => "Day is outside range!",
            &DateError::InvalidMonth(_) => "Month is outside range!",
        }
    }

    // cause method returns None by default
}

Opmerking: in het algemeen moet Option<T> niet worden gebruikt voor het melden van fouten. Option<T> geeft een verwachte mogelijkheid aan dat een waarde niet bestaat en een enkele duidelijke reden daarvoor. Result<T, E> wordt daarentegen gebruikt om onverwachte fouten tijdens de uitvoering te melden, en met name wanneer er meerdere storingsmodi zijn om ze te onderscheiden. Verder wordt Result<T, E> alleen gebruikt als retourwaarden. ( Een oude discussie. )



Modified text is an extract of the original Stack Overflow Documentation
Licentie onder CC BY-SA 3.0
Niet aangesloten bij Stack Overflow