我有大约10个结构,每个结构之间有5-10个字段,我希望能够使用相同的格式将它们打印出来。
我的大多数结构看起来像这样:
struct Example {
a: Option<String>,
b: Option<i64>,
c: Option<String>,
... etc
}
[我希望能够为impl
定义一个fmt::Display
,而不必再次枚举这些字段,因此,如果添加了新字段,则不会丢失任何一个。
对于结构:
let eg = Example{
a: Some("test".to_string),
b: Some(123),
c: None,
}
我想要输出格式:
a: test
b: 123
c: -
我目前正在使用#[derive(Debug)]
,但我不喜欢它打印出Some(X)
和None
以及其他一些信息。
如果我知道结构中的所有值都是Option<T: fmt::Display>
,是否可以自己生成Display
方法而不必再次列出字段?
这可能不是最基本的实现,但是您可以派生可序列化并使用serde
条板箱。以下是自定义序列化程序的示例:https://serde.rs/impl-serializer.html
在您的情况下,它可能要简单得多(您只需要少数几种类型,并且可以对任何意外情况进行惊慌/忽略)。
另一种方法可能是编写宏并创建自己的轻量级序列化解决方案。
我最终用一个宏解决了这个问题。虽然这不是理想的方法,但可以完成工作。
我的宏当前看起来像这样:
macro_rules! MyDisplay {
($struct:ident {$( $field:ident:$type:ty ),*,}) => {
#[derive(Debug)]
pub struct $struct { pub $($field: $type),*}
impl fmt::Display for $struct {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
$(
write!(f, "{}: {}\n",
stringify!($field).to_string(),
match &self.$field {
None => "-".to_string(),
Some(x) => format!("{:#?}", x)
}
)?;
)*
Ok(())
}
}
};
}
可以这样使用:
MyDisplay! {
Example {
a: Option<String>,
b: Option<i64>,
c: Option<String>,
}
}
我的宏基于Cerberus在此提供的https://stackoverflow.com/a/54177889/1355121宏