Как я могу сериализовать перечисление, не включая имя варианта перечисления?

Я пытаюсь сериализовать перечисление в строку JSON. Я реализовал Serialize черта моего перечисления, как это описано в документации, но я всегда получаю {"offset":{"Int":0}} вместо желаемого {"offset":0},

extern crate serde;
extern crate serde_json;

use std::collections::HashMap;

use serde::ser::{Serialize, Serializer};

#[derive(Debug)]
enum TValue<'a> {
    String(&'a str),
    Int(&'a i32),
}

impl<'a> Serialize for TValue<'a> {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        match *self {
            TValue::String(ref s) => serializer.serialize_newtype_variant("TValue", 0, "String", s),
            TValue::Int(i) => serializer.serialize_newtype_variant("TValue", 1, "Int", &i),
        }
    }
}

fn main() {
    let offset: i32 = 0;
    let mut request_body = HashMap::new();
    request_body.insert("offset", TValue::Int(&offset));
    let serialized = serde_json::to_string(&request_body).unwrap();
    println!("{}", serialized); // {"offset":{"Int":0}}
}

1 ответ

Решение

Вы можете использовать untagged атрибут, который будет производить желаемый результат. Вам не нужно будет реализовывать Serialize себя с этим:

#[derive(Debug, Serialize)]
#[serde(untagged)]
enum TValue<'a> {
    String(&'a str),
    Int(&'a i32),
}

Если вы хотели реализовать Serialize я полагаю, что вы хотите пропустить свой вариант, поэтому не стоит использовать serialize_newtype_variant() как это выставляет ваш вариант. Вы должны использовать serialize_str() а также serialize_i32() непосредственно:

impl<'a> Serialize for TValue<'a> {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        match *self {
            TValue::String(s) => serializer.serialize_str(s),
            TValue::Int(i) => serializer.serialize_i32(*i),
        }
    }
}

Он производит желаемый результат:

{"offset":0}
Другие вопросы по тегам