问题:
我有一个包含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);
}
}