design-patterns - SNAFU 库使用什么设计模式来扩展外部类型?

标签 design-patterns rust idioms

我一直在研究有趣的 SNAFU library .

SNAFU 页面中的一个稍微修改过的独立示例如下:

use snafu::{ResultExt, Snafu};
use std::{fs, io, path::PathBuf};

#[derive(Debug, Snafu)]
enum Error {
    #[snafu(display("Unable to read configuration from {}: {}", path.display(), source))]
    ReadConfiguration { source: io::Error, path: PathBuf },
}

type Result<T, E = Error> = std::result::Result<T, E>;

fn process_data() -> Result<()> {
    let path = "config.toml";
    let read_result: std::result::Result<String, io::Error> = fs::read_to_string(path);    
    let _configuration = read_result.context(ReadConfiguration { path })?;
    Ok(())
}

fn main() {
    let foo = process_data();

    match foo {
        Err(e) => println!("Hello {}", e),
        _ => println!("success")
    }
}

我所做的更改是使 fs::read_to_string(path)Result 的类型在 process_data() 中显式>.

鉴于此,我不明白 read_result 如何有可用的 context 方法,因为 std::result::Result 文档不对上下文进行任何引用(如果您删除 SNAFU 内容并尝试访问上下文,编译器同样会提示)。

这里使用的模式对我来说并不明显。我天真的理解是,由于孤立规则,外部类型无法扩展,但这里发生的事情看起来很像这样的扩展。

我也对 type Result... 行感到困惑。我知道类型别名,但没有使用左侧分配了泛型的语法。显然,这是设计模式的重要组成部分。

我的要求是澄清这里使用的是什么模式以及它是如何工作的。它似乎触及了 Rust 的一些非常有趣的方面。进一步阅读将是有值(value)的!

最佳答案

ResultExt 是一个扩展特征,这就是为什么它使用有点常见的后缀 Ext .

减少,实现包含特征的定义和特定类型(或仅一个)的特征的一小部分实现:

pub trait ResultExt<T, E>: Sized {
    fn context<C, E2>(self, context: C) -> Result<T, E2>
    where
        C: IntoError<E2, Source = E>,
        E2: std::error::Error + ErrorCompat;
}

impl<T, E> ResultExt<T, E> for std::result::Result<T, E> {
    fn context<C, E2>(self, context: C) -> Result<T, E2>
    where
        C: IntoError<E2, Source = E>,
        E2: std::error::Error + ErrorCompat,
    {
        self.map_err(|error| context.into_error(error))
    }
}

通过导入ResultExt ,您将特征及其方法带入范围。图书馆已为 Result 实现了它们类型,因此您也可以使用它们。

另见:

I'm aware of type aliasing, but not using the syntax in which the left hand side has a generic assigned. Clearly, this is an important part of the design pattern.

使用 .context() 的能力并不重要,这只是我鼓励的一种做法。最定制Result别名隐藏名称 Result (例如 std::io::Result ),这意味着如果您需要使用不同的错误类型,则需要使用丑陋的完整路径或其他别名(例如 type StdResult<T, E> = std::result::Result<T, E> )。

通过创建本地别名 Result使用默认通用,用户可以键入Result<T>而不是更冗长的Result<T, MyError> , 但仍然可以使用 Result<T, SomeOtherError>需要的时候。有时,我什至会更进一步,为成功类型定义一个默认类型。这在单元测试中最常见:

mod test {
    type Result<T = (), E = Box<dyn std::error::Error>> = std::result::Result<T, E>;

    fn setup() -> Result<String> {
        Ok(String::new())
    }

    fn special() -> Result<String, std::io::Error> {
        std::fs::read_to_string("/etc/hosts")
    }

    #[test]
    fn x() -> Result {
        Ok(())
    }
}

关于design-patterns - SNAFU 库使用什么设计模式来扩展外部类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57995352/

相关文章:

当您发现自己需要按特定顺序执行事件时的 C# 设计模式

java - 如何在 Java 中反转 int 数组?

java - Scala 的哪些特性无法转换为 Java?

docker - 在 docker 中构建 rust 项目导致 Cargo 卡住下载依赖项

methods - 如何为此Rust特征使用相同的默认实现

rust - 使用 nalgebra 将矩阵乘以向量并进行通用大小调整时出错

haskell - 在 Haskell 中生成有限的素数列表

c++ - CRTP、cpp 文件中的前向声明和模板

php - setter/getter 中重复开关的设计模式?

java - 几个枚举如何扩展一个包含用于 Java 中这些枚举的方法的类?