32

我有一个用 C# 定义的枚举,我将它的值存储为字符,如下所示:

public enum CardType
{
    Artist = 'A',
    Contemporary = 'C',
    Historical = 'H',
    Musician = 'M',
    Sports = 'S',
    Writer = 'W'
}

我正在尝试使用 JSON.NET 进行反序列化,但传入的 JSON 是使用 CHAR 值(字符串)而不是枚举的 int 值编写的,如下所示:

[{"CardType","A"},{"CardType", "C"}]

是否可以定义某种转换器,允许我手动将字符解析为枚举值?

我尝试创建一个 JsonConverter,但不知道该怎么做,同时仅将它应用于此属性而不是整个解析的对象。这是我尝试过的:

public class EnumerationConverter : JsonConverter
{
    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        serializer.Serialize(writer, value);
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        if (reader.TokenType == JsonToken.Null)
        {
            return null;
        }

        int value = serializer.Deserialize<int>(reader);
        return (CardType)value;
    }

    public override bool CanConvert(Type objectType)
    {
        return objectType.IsSubclassOf(typeof(string));
    }
}

逻辑可能是错误的,我可以解决这个问题,但问题是根本没有调用 ReadJson()。

CanConvert 是,但它似乎是为每个属性调用的,而不仅仅是我为它定义的一个属性:

public class Card
{
            private CardType type;
        [JsonConverter(typeof(EnumerationConverter))]
        public CardType Type
        {
            get { return type; }
            set { type = value; }
        }
}

我确定我做错了,但是找不到有关如何为单个字段执行此操作的文档...

我错过了什么?

4

4 回答 4

47

您不需要自定义JsonConverter,您可以将内置StringEnumConverterEnumMemberAttribute(来自System.Runtime.Serialization程序集)的组合一起使用。

如果没有EnumMemberAttribute它,它会使用枚举名称,因此 Artist、Contemporary 等,因此您需要将其名称更改为您的 A、C 等值。

但这不是最好的解决方案,因为您必须重复两次您的值,但它有效:

[JsonConverter(typeof(StringEnumConverter))]
public enum CardType
{
    [EnumMember(Value = "A")]
    Artist = 'A',
    [EnumMember(Value = "C")]
    Contemporary = 'C',
    [EnumMember(Value = "H")]
    Historical = 'H',
    [EnumMember(Value = "M")]
    Musician = 'M',
    [EnumMember(Value = "S")]
    Sports = 'S',
    [EnumMember(Value = "W")]
    Writer = 'W'
}
于 2013-08-31T18:54:16.963 回答
7

此代码完美运行:

CardType[] array = { CardType.Artist, CardType.Contemporary };
string s = JsonConvert.SerializeObject(array);
var array2 = JsonConvert.DeserializeObject<CardType[]>(s);

更新
开箱即用怎么样StringEnumConverter

[JsonConverter(typeof(StringEnumConverter))]
public CardType Type { get; set; }
于 2013-08-31T18:47:30.217 回答
2

您只需添加 SerializerSettings.Converters.Add(new StringEnumConverter());

到您的 BrowserJsonFormatter 类

public class BrowserJsonFormatter : JsonMediaTypeFormatter
{
    public BrowserJsonFormatter()
    {
        SupportedMediaTypes.Add(new MediaTypeHeaderValue("text/html"));
        SerializerSettings.Formatting = Formatting.Indented;
        SerializerSettings.NullValueHandling = NullValueHandling.Ignore;
        SerializerSettings.ContractResolver = new CamelCasePropertyNamesContractResolver();
        SerializerSettings.Converters.Add(new EmptyToNullConverter());
        SerializerSettings.Converters.Add(new StringEnumConverter());
        //SerializerSettings.DefaultValueHandling = DefaultValueHandling.IgnoreAndPopulate;
    }

    public override void SetDefaultContentHeaders(Type type, HttpContentHeaders headers, MediaTypeHeaderValue mediaType)
    {
        base.SetDefaultContentHeaders(type, headers, mediaType);
        headers.ContentType = new MediaTypeHeaderValue("application/json");
    }
}
于 2017-01-26T19:49:31.103 回答
0

这是使用自定义 JsonConverter 对我有用的方法。我白天是一名 Java 开发人员,所以也许专业的 C# 开发人员可以改进这一点或发现任何问题。

public class CharEnumConverter<T> : JsonConverter<T>
{
    public override void WriteJson(JsonWriter writer, T value, JsonSerializer serializer)
    {
        char c = (char)(int)Enum.Parse(typeof(T), value.ToString());
        writer.WriteValue(c + "");
    }

    public override T ReadJson(JsonReader reader, Type objectType, T existingValue, bool hasExistingValue, JsonSerializer serializer)
    {
        string stringValue = (string)reader.Value;
        char charValue = stringValue[0];
        int intValue = (int)charValue;
        string intValueString = intValue + "";

        if (Enum.IsDefined(typeof(T), intValue)) {
            T result = (T)Enum.Parse(typeof(T), intValueString);
            return result;
        } else {
            throw new Exception("Char value [" + charValue + "] is not defined for Enum [" + typeof(T).Name + "]");
        }
    }
}

我想作为 Json 读/写的枚举的示例用法。

[CharEnum]    
public enum MyEnum {
    NONE = 'A',
    YAY = 'B'
}

public class MyClass {

    [JsonConverter(typeof(CharEnumConverter<MyEnum>))]    
    public MyEnum myfield;

}

它将 JSON 字符串中的“A”读取到值为 NONE 的枚举字段中。它将一个值为 YAY 的枚举字段写入 JSON 字符串为“B”

于 2021-06-15T05:53:16.803 回答