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

Я пытаюсь сериализовать перечисление в строку 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}}
}

person Jade    schedule 27.08.2018    source источник


Ответы (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}
person Stargateur    schedule 27.08.2018