rust - Visitor 特性如何只允许部分实现?

标签 rust visitor-pattern serde

在我对 Rust 的有限理解中,我认为 trait 就像接口(interface)——当你实现一个接口(interface)时,你需要实现所有的方法。

我现在正在编写自定义 serde::Deserializer。

https://docs.serde.rs/serde/de/trait.Visitor.html#example

/// A visitor that deserializes a long string - a string containing at least
/// some minimum number of bytes.
struct LongString {
    min: usize,
}

impl<'de> Visitor<'de> for LongString {
    type Value = String;

    fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
        write!(formatter, "a string containing at least {} bytes", self.min)
    }

    fn visit_str<E>(self, s: &str) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        if s.len() >= self.min {
            Ok(s.to_owned())
        } else {
            Err(de::Error::invalid_value(Unexpected::Str(s), &self))
        }
    }
}

https://serde.rs/impl-deserialize.html

use std::fmt;

use serde::de::{self, Visitor};

struct I32Visitor;

impl<'de> Visitor<'de> for I32Visitor {
    type Value = i32;

    fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
        formatter.write_str("an integer between -2^31 and 2^31")
    }

    fn visit_i8<E>(self, value: i8) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(i32::from(value))
    }

    fn visit_i32<E>(self, value: i32) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(value)
    }

    fn visit_i64<E>(self, value: i64) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        use std::i32;
        if value >= i64::from(i32::MIN) && value <= i64::from(i32::MAX) {
            Ok(value as i32)
        } else {
            Err(E::custom(format!("i32 out of range: {}", value)))
        }
    }

    // Similar for other methods:
    //   - visit_i16
    //   - visit_u8
    //   - visit_u16
    //   - visit_u32
    //   - visit_u64
}

Visitor trait 似乎很神奇——您只需要实现您想要实现的方法,而其他方法默认为某种形式的错误。

这个魔法是如何工作的?

最佳答案

特征方法可以有默认实现。具有默认实现的方法在文档中称为“提供的方法”,它们可以选择性地被实现覆盖。

你可以看到 default implementations for the Visitor trait in the source code .并非所有的都返回错误,但有些也有合理的默认实现,例如较小的数字类型遵循 visit_i64()visit_u64(),因此您只需定义这两种方法即可获得所有整数类型的合理实现。

关于rust - Visitor 特性如何只允许部分实现?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63847222/

相关文章:

syntax - 如何部分初始化 ArrayVec?

java - 关于访问者模式的问题(Java 示例)

java - 将表示数学表达式的树转换为没有多余括号的字符串

rust - 如何在 Rust 的稳定版本上使用 serde

json - 无法使用本地创建的向量,因为 "borrowed"

rust - GTK4 : How to specify rows and columns of a GtkGrid object within a GtkBuilder ui file

rust - 检查自定义结构的相等性

functional-programming - 如何在没有 for 循环的情况下搜索二维数组?

c++ - 使用 boost::bind 和重载函数的访问者模式

rust - 如何忽略序列的部分项