【问题标题】:Deserializing TOML into vector of enum with values将 TOML 反序列化为具有值的枚举向量
【发布时间】:2018-02-06 11:06:15
【问题描述】:

我正在尝试读取一个 TOML 文件以创建一个结构,该结构包含一个带有关联值的枚举向量。这是示例代码:

extern crate serde;
#[macro_use]
extern crate serde_derive;
extern crate toml;

use std::fs::File;
use std::io::Read;

#[derive(Debug, Deserialize, PartialEq)]
struct Actor {
    name: String,
    actions: Vec<Actions>,
}

#[derive(Debug, Deserialize, PartialEq)]
enum Actions {
    Wait(usize),
    Move { x: usize, y: usize },
}

fn main() {
    let input_file = "./sample_actor.toml";
    let mut file = File::open(input_file).unwrap();
    let mut file_content = String::new();
    let _bytes_read = file.read_to_string(&mut file_content).unwrap();
    let actor: Actor = toml::from_str(&file_content).unwrap();
    println!("Read actor {:?}", actor);
}

Cargo.toml

[dependencies]
serde_derive = "1.0.10"
serde = "1.0.10"
toml = "0.4.2"

sample_actor.toml

name = "actor1"
actions = [move 3 4, wait 20, move 5 6]

我知道文件看起来“错误”,但我不知道应该如何在 TOML 文件中编写操作,以便 crate 能够将它们识别为具有 X 个值的枚举。

使用cargo run 运行此示例时出现以下错误:

thread 'main' panicked at 'called `Result::unwrap()` on an `Err` value: Error { inner: ErrorInner { kind: NumberInvalid, line: Some(1), col: 11, message: "", key: [] } }', /checkout/src/libcore/result.rs:906:4
note: Run with `RUST_BACKTRACE=1` for a backtrace.

我知道我可能需要为我的枚举实现FromStr 以将字符串转换为我的枚举,并且我简要地知道可以实现自定义反序列化器以以特定方式反序列化,但我不确定这些是如何实现的拼凑在一起。

似乎使用 serde_json 而不是 TOML 的等效示例可以直接工作(当然使用 JSON 文件而不是 TOML)。

JSON 版本的代码:

extern crate serde;
extern crate serde_json;

#[macro_use]
extern crate serde_derive;

use serde_json::Error;
use std::fs::File;
use std::io::Read;

#[derive(Debug, Serialize, Deserialize)]
enum Foo {
    bar(u32),
    baz { x: u32, y: u32 },
}

#[derive(Debug, Serialize, Deserialize)]
struct Address {
    street: String,
    city: String,
    nums: Vec<Foo>,
}

fn main() {
    /*
        let address = Address {
            street: "10 Downing Street".to_owned(),
            city: "London".to_owned(),
            nums : vec![Foo::bar(1), Foo::baz{x : 2, y : 3}],
        };

        // Serialize it to a JSON string.
        let j = serde_json::to_string(&address).unwrap();

        // Print, write to a file, or send to an HTTP server.
        println!("{}", j);
    */
    let input_file = "./sample_address.json";
    let mut file = File::open(input_file).unwrap();
    let mut file_content = String::new();
    let _bytes_read = file.read_to_string(&mut file_content).unwrap();
    let address: Address = serde_json::from_str(&file_content).unwrap();
    println!("{:?}", address);
}

本例中读取/写入的 JSON 数据为:

Address { street: "10 Downing Street", city: "London", nums: [bar(1), baz { x: 2, y: 3 }] }

也许 TOML crate 不支持我的用例?

【问题讨论】:

  • 找出你的 toml 应该是什么样子的最快方法是派生 Serialize 并为你的结构发出 toml。
  • @oli_obk-ker 这真是个好主意,谢谢!
  • 既然你提到了它,它似乎很明显,我觉得一开始就没有尝试这个很愚蠢哈哈。
  • 好吧,看来 toml crate 有同样的问题:你如何序列化枚举?当我创建一个演员并在其上使用 toml::to_string() 时,它会产生输出 Err(UnsupportedType)。如果我更改 actor 的定义,使动作为 Vec 而不是 Vec,它会产生有效的 TOML。

标签: rust serde toml


【解决方案1】:

Serde 有lots of options for serializing enums。适合您的情况:

use serde::{Deserialize, Serialize}; // 1.0.117
use toml; // 0.5.7

#[derive(Debug, Serialize, Deserialize, PartialEq)]
#[serde(tag = "type", content = "args")]
enum Actions {
    Wait(usize),
    Move { x: usize, y: usize },
}

fn main() {
    let a_wait = Actions::Wait(5);
    println!("{}", toml::to_string(&a_wait).unwrap());

    let a_move = Actions::Move { x: 1, y: 1 };
    println!("{}", toml::to_string(&a_move).unwrap());
}
type = "Wait"
args = 5
type = "Move"

[args]
x = 1
y = 1

【讨论】:

  • 谢谢,谢普。那个 serde 枚举文档正是我想要的。干杯!
猜你喜欢
  • 1970-01-01
  • 2016-01-28
  • 2016-10-16
  • 1970-01-01
  • 1970-01-01
  • 2013-09-04
  • 1970-01-01
  • 2018-10-11
  • 1970-01-01
相关资源
最近更新 更多