8

Json.NETシリアライザーを作成して、IDictionary<,>インスタンスをキー/値プロパティを持つオブジェクトの配列にシリアル化するにはどうすればよいですか? デフォルトでは、Key の値を JSON オブジェクトのプロパティ名にシリアル化します。

基本的に、次のようなものが必要です。

[{"key":"some key","value":1},{"key":"another key","value":5}]

それ以外の:

{{"some key":1},{"another key":5}}

シリアライザーの設定に追加しようとしましKeyValuePairConverterたが、効果がありません。(このコンバーターは のタイプでは無視されることがわかりましたIDictionary<>が、他のライブラリから受け取ったオブジェクトのタイプを簡単に変更できないため、 からIDictionary<>への変更ICollection<KeyValuePair<>>はオプションではありません。)

4

2 に答える 2

5

このコンバーターを機能させることができました。

using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using Newtonsoft.Json;
using Newtonsoft.Json.Converters;

public class CustomDictionaryConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return (typeof(IDictionary).IsAssignableFrom(objectType) || 
                TypeImplementsGenericInterface(objectType, typeof(IDictionary<,>)));
    }

    private static bool TypeImplementsGenericInterface(Type concreteType, Type interfaceType)
    {
        return concreteType.GetInterfaces()
               .Any(i => i.IsGenericType && i.GetGenericTypeDefinition() == interfaceType);
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        Type type = value.GetType();
        IEnumerable keys = (IEnumerable)type.GetProperty("Keys").GetValue(value, null);
        IEnumerable values = (IEnumerable)type.GetProperty("Values").GetValue(value, null);
        IEnumerator valueEnumerator = values.GetEnumerator();

        writer.WriteStartArray();
        foreach (object key in keys)
        {
            valueEnumerator.MoveNext();

            writer.WriteStartObject();
            writer.WritePropertyName("key");
            writer.WriteValue(key);
            writer.WritePropertyName("value");
            serializer.Serialize(writer, valueEnumerator.Current);
            writer.WriteEndObject();
        }
        writer.WriteEndArray();
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        throw new NotImplementedException();
    }
}

コンバーターの使用例を次に示します。

IDictionary<string, int> dict = new Dictionary<string, int>();
dict.Add("some key", 1);
dict.Add("another key", 5);

string json = JsonConvert.SerializeObject(dict, new CustomDictionaryConverter());
Console.WriteLine(json);

そして、これが上記の出力です:

[{"key":"some key","value":1},{"key":"another key","value":5}]
于 2013-08-22T16:59:13.043 に答える
2

別の方法を考え出しました - カスタムの ContractResolver を作成し、それをJsonSerializerSettings(デ) シリアル化の前に設定できます。CamelCasePropertyNamesContractResolver以下は、シリアル化されたプロパティ名をキャメルケースに変換する組み込みから派生したものDefaultContractResolverですが、名前を変更したくない場合は、から派生させることができます。

public class DictionaryFriendlyContractResolver : CamelCasePropertyNamesContractResolver
{
    protected override JsonContract CreateContract(Type objectType)
    {
        if (objectType.IsGenericType && objectType.GetGenericTypeDefinition() == typeof(IDictionary<,>))
            return new JsonArrayContract(objectType);
        if (objectType.GetInterfaces().Any(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IDictionary<,>)))
            return new JsonArrayContract(objectType);
        return base.CreateContract(objectType);
    }
}

使用法:

var cfg = new JsonSerializerSettings();
cfg.ContractResolver = new DictionaryFriendlyContractResolver();
string json = JsonConvert.SerializeObject(myModel, cfg);
于 2013-08-23T14:55:59.557 に答える