在 C# 中使用枚举值反序列化字典<字符串、对象>

本文关键字:字典 字符串 对象 反序列化 枚举 | 更新日期: 2023-09-27 17:56:42

我正在尝试在 C# 中序列化/反序列化Dictionary<string, object>。对象可以是可序列化的任何内容。

Json.NET 几乎可以工作,但是如果字典中的值是enum,则反序列化不正确,因为它被反序列化为longTypeNameHandling.All没有任何区别。

序列化库还有其他快速解决方案吗?结果不必是 JSON,但必须是文本。

我对传递给字典的数据也没有影响。我只需要序列化和反序列化任何进入我的方式。

编辑:StringEnumConverter没有帮助。数据被转换回Dictionary<string, object>,所以反序列化器不知道序列化的值是一个enum。它将其视为一个对象,StringEnumConverter它在反序列化时仍然是string;它被反序列化为没有转换器的long。JSON.NET 不会保留枚举。

我想提供的解决方案是现有接口的实现,该接口被注入到我无法更改的现有解决方案中。

编辑2:这是我正在尝试做的事情的一个例子

public enum Foo { A, B, C }
public enum Bar { A, B, C }
public class Misc { public Foo Foo { get; set; } }

var dict = new Dictionary<string, object>();
dict.Add("a", Foo.A);
dict.Add("b", Bar.B);
dict.Add("c", new Misc());
// serialize dict to a string s
// deserialize s to a Dictionary<string, object> dict2
Assert.AreEqual(Foo.A, dict2["a"]);
Assert.AreEqual(Bar.B, dict2["b"]);

重要提示:我无法控制dict;它实际上是一个派生自Dictionary<string, object>的自定义类型:我只需要确保反序列化时所有键和值都来自同一类型,这样就不需要强制转换。再说一次,我不必使用 JSON;也许还有其他一些序列化程序可以处理这项工作!?

在 C# 中使用枚举值反序列化字典<字符串、对象>

大概你已经在用 TypeNameHandling.All 序列化字典了,它应该通过发出一个 "$type" 对象属性以及对象本身来正确序列化和反序列化new Misc()值。 不幸的是,对于枚举等类型(以及其他如 intlong ),这不起作用,因为它们被序列化为 JSON 基元,没有机会包含 "$type" 属性。

解决方案是在序列化具有object值的字典时,按照本答案的行序列化可以封装类型信息的基元值的对象包装器。 由于您无法修改任何传入对象,并且需要"注入"正确的包装器,因此您可以使用自定义协定解析程序执行此操作,该解析程序将适当的项目转换器应用于字典值:

public class UntypedToTypedValueContractResolver : DefaultContractResolver
{
    // As of 7.0.1, Json.NET suggests using a static instance for "stateless" contract resolvers, for performance reasons.
    // http://www.newtonsoft.com/json/help/html/ContractResolver.htm
    // http://www.newtonsoft.com/json/help/html/M_Newtonsoft_Json_Serialization_DefaultContractResolver__ctor_1.htm
    // "Use the parameterless constructor and cache instances of the contract resolver within your application for optimal performance."
    // See also https://stackoverflow.com/questions/33557737/does-json-net-cache-types-serialization-information
    static UntypedToTypedValueContractResolver instance;
    // Explicit static constructor to tell C# compiler not to mark type as beforefieldinit
    static UntypedToTypedValueContractResolver() { instance = new UntypedToTypedValueContractResolver(); }
    public static UntypedToTypedValueContractResolver Instance { get { return instance; } }
    protected override JsonDictionaryContract CreateDictionaryContract(Type objectType)
    {
        var contract = base.CreateDictionaryContract(objectType);
        if (contract.DictionaryValueType == typeof(object) && contract.ItemConverter == null)
        {
            contract.ItemConverter = new UntypedToTypedValueConverter();
        }
        return contract;
    }
}
class UntypedToTypedValueConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        throw new NotImplementedException("This converter should only be applied directly via ItemConverterType, not added to JsonSerializer.Converters");
    }
    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        if (reader.TokenType == JsonToken.Null)
            return null;
        var value = serializer.Deserialize(reader, objectType);
        if (value is TypeWrapper)
        {
            return ((TypeWrapper)value).ObjectValue;
        }
        return value;
    }
    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        if (serializer.TypeNameHandling == TypeNameHandling.None)
        {
            Debug.WriteLine("ObjectItemConverter used when serializer.TypeNameHandling == TypeNameHandling.None");
            serializer.Serialize(writer, value);
        }
        // Handle a couple of simple primitive cases where a type wrapper is not needed
        else if (value is string)
        {
            writer.WriteValue((string)value);
        }
        else if (value is bool)
        {
            writer.WriteValue((bool)value);
        }
        else
        {
            var contract = serializer.ContractResolver.ResolveContract(value.GetType());
            if (contract is JsonPrimitiveContract)
            {
                var wrapper = TypeWrapper.CreateWrapper(value);
                serializer.Serialize(writer, wrapper, typeof(object));
            }
            else
            {
                serializer.Serialize(writer, value);
            }
        }
    }
}
public abstract class TypeWrapper
{
    protected TypeWrapper() { }
    [JsonIgnore]
    public abstract object ObjectValue { get; }
    public static TypeWrapper CreateWrapper<T>(T value)
    {
        if (value == null)
            return new TypeWrapper<T>();
        var type = value.GetType();
        if (type == typeof(T))
            return new TypeWrapper<T>(value);
        // Return actual type of subclass
        return (TypeWrapper)Activator.CreateInstance(typeof(TypeWrapper<>).MakeGenericType(type), value);
    }
}
public sealed class TypeWrapper<T> : TypeWrapper
{
    public TypeWrapper() : base() { }
    public TypeWrapper(T value)
        : base()
    {
        this.Value = value;
    }
    public override object ObjectValue { get { return Value; } }
    public T Value { get; set; }
}

然后像这样使用它:

var settings = new JsonSerializerSettings
{
    TypeNameHandling = TypeNameHandling.All,
    ContractResolver = UntypedToTypedValueContractResolver.Instance,
    Converters = new [] { new StringEnumConverter() }, // If you prefer
};
var json = JsonConvert.SerializeObject(dict, Formatting.Indented, settings);
var dict2 = JsonConvert.DeserializeObject<Dictionary<string, object>>(json, settings);

样品小提琴。

最后,在使用 TypeNameHandling 时,请注意 Newtonsoft 文档中的以下警告:

当应用程序从外部源反序列化 JSON 时,应谨慎使用类型名称处理。使用非 None 的值反序列化时,应使用自定义序列化绑定程序验证传入类型。

有关为什么可能需要这样做的讨论,请参阅 Newtonsoft Json 中的 TypeNameHandling 注意事项

假设您有权修改对象类,则可以将 JsonCoverter 属性添加到类的枚举成员中。

[JsonConverter(typeof(StringEnumConverter))]