使用具有 IDictionary<Mycustomclass、List 的自定义解析程序的反序列化问题<string>>



问题:

我有一个包含IDictionary<MyCustomClass, List<string>>的类Foo。我使用自定义ContractResolver,如本答案所示,以便我的字典将序列化为包含"键"one_answers"值"属性的对象数组。这部分很好。但是,当我尝试使用相同的解析器将JSON反序列化回Foo时,我得到如下所示的错误。

错误:

运行时异常(第46行):无法将当前JSON数组(例如[1,2,3])反序列化为类型System. collections . generic . dictionary 2[mycustomclass,System.Collections.Generic.List [System. collections . generic . dictionary]字符串]]'因为该类型需要一个JSON对象(例如{"name":"value"})来反序列化正确。

要修复此错误,要么将JSON更改为JSON对象(例如{"name":"value"}),要么将反序列化类型更改为数组或a实现集合接口的类型(例如ICollection, IList)比如可以从JSON数组反序列化的List。也可以将JsonArrayAttribute添加到类型中以强制它从JSON数组反序列化。

路径'Dict2',第12行,位置13。

Dotnetfiddle:

https://dotnetfiddle.net/BKaKab

我的代码:

using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using Newtonsoft.Json;
using Newtonsoft.Json.Serialization;

public class mycustomclass
{
    public string name {get;set;}
    public List<string> Productlist {get;set;}
    public List<string> SelectedItems{get;set;}
}

public class Program
{
    public static void Main()
    {
        mycustomclass omycustomclass = new mycustomclass{name = "Test",Productlist =  new string[]{"Product1","Product2","Product3"}.ToList(), SelectedItems = new string[]{"Item1","Item2","Item3"}.ToList()};
        mycustomclass omycustomclass2 = new mycustomclass{name = "Test",Productlist =  new string[]{"Product4","Product5","Product6"}.ToList(), SelectedItems = new string[]{"Item4","Item5","Item6"}.ToList()};
        Foo foo = new Foo();
        foo.Dict = new Dictionary<string, string>();
        foo.Dict.Add("Gee", "Whiz");
        foo.Dict.Add("Fizz", "Bang");
        Dictionary<mycustomclass, List<string>> custom = new Dictionary<mycustomclass, List<string>>();     
        custom.Add(omycustomclass, new string[]{"l1","l2","l3"}.ToList());
        custom.Add(omycustomclass2, new string[]{"l4","l5","l6"}.ToList());
        foo.Dict2 = custom;

        JsonSerializerSettings settings = new JsonSerializerSettings();
        settings.Formatting = Formatting.Indented;
        settings.ContractResolver = new DictionaryAsArrayResolver();
        // serialize
        string json = JsonConvert.SerializeObject(foo, settings);
        Console.WriteLine(json);
        Console.WriteLine();
        // deserialize
        foo = JsonConvert.DeserializeObject<Foo>(json, settings);
        foreach (var kvp in foo.Dict)
        {
            Console.WriteLine(kvp.Key + ": " + kvp.Value);
        }
    }
    class Foo
    {
        public Dictionary<string, string> Dict { get; set; }
        public IDictionary<mycustomclass, List<string>> Dict2 { get; set; }
    }
}
class DictionaryAsArrayResolver : DefaultContractResolver
{
    protected override JsonContract CreateContract(Type objectType)
    {
        if (objectType.GetInterfaces().Any(i => i == typeof(IDictionary) || 
            (i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IDictionary<,>))))
        {
            return base.CreateArrayContract(objectType);
        }
        return base.CreateContract(objectType);
    }
}
JSON:

{
  "Dict": [
    {
      "Key": "Gee",
      "Value": "Whiz"
    },
    {
      "Key": "Fizz",
      "Value": "Bang"
    }
  ],
  "Dict2": [
    {
      "Key": {
        "name": "Test",
        "Productlist": [
          "Product1",
          "Product2",
          "Product3"
        ],
        "SelectedItems": [
          "Item1",
          "Item2",
          "Item3"
        ]
      },
      "Value": [
        "l1",
        "l2",
        "l3"
      ]
    },
    {
      "Key": {
        "name": "Test",
        "Productlist": [
          "Product1",
          "Product2",
          "Product3"
        ],
        "SelectedItems": [
          "Item1",
          "Item2",
          "Item3"
        ]
      },
      "Value": [
        "l1",
        "l2",
        "l3"
      ]
    }
  ]
}

谁能帮我把它修好吗?

不确定这是否是最好的方法,但它解决了你的问题。创建一个转换器

internal class MyConverter : CustomCreationConverter<IDictionary<CustomClass, List<String>>>
{
    public override IDictionary<CustomClass, List<String>> Create(Type objectType)
    {
        return new Dictionary<CustomClass, List<String>>();
    }
    public override bool CanConvert(Type objectType)
    {
        return objectType == typeof (object) || base.CanConvert(objectType);
    }
}

添加到你的设置:

var settings = new JsonSerializerSettings
    {
        Formatting = Formatting.Indented,
        ContractResolver = new DictionaryAsArrayResolver(),
        Converters = new JsonConverter[] {new MyConverter()}
    };

示例

更简洁的方法是通过设置JsonArrayContract来定义创建方法。OverrideCreator:

public class DictionaryAsArrayResolver : DefaultContractResolver
{
    public override JsonContract ResolveContract(Type objectType)
    {
        if (IsDictionary(objectType))
        {
            JsonArrayContract contract = base.CreateArrayContract(objectType);
            contract.OverrideCreator = (args) => CreateInstance(objectType);
            return contract;
        }
        return base.CreateContract(objectType);
    }
    internal static bool IsDictionary(Type objectType)
    {
        if (objectType.IsGenericType && objectType.GetGenericTypeDefinition() == typeof(IDictionary<,>))
        {
            return true;
        }
        if (objectType.GetInterface(typeof(IDictionary<,>).Name) != null)
        {
            return true;
        }
        return false;
    }
    private object CreateInstance(Type objectType)
    {
        Type dictionaryType = typeof(Dictionary<,>).MakeGenericType(objectType.GetGenericArguments());
        return Activator.CreateInstance(dictionaryType);
    }
}

最新更新