37

Json.Net 通常将 a 序列Dictionary<k,v>化为一个集合;

"MyDict": {
  "Apples": {
    "Taste": 1341181398,
    "Title": "Granny Smith",
  },
  "Oranges": {
    "Taste": 9999999999,
    "Title": "Coxes Pippin",
  },
 }

这是伟大的。从 SO 上环顾四周,这似乎是大多数人想要的。但是,在这种特殊情况下,我想在 myDictionary<k,v>和 Array 格式之间进行序列化;

"MyDict": [
    "k": "Apples",
    "v": {
        "Taste": 1341181398,
        "Title": "Granny Smith",
    }
  },
    "k:": "Oranges",
    "v:": {
        "Taste": 9999999999,
        "Title": "Coxes Pippin",
    }
  },
]

有没有一种简单的方法可以用我现有的字段类型来做到这一点?例如,是否有我可以注释的属性?

4

6 回答 6

45

实现此目的的另一种方法是使用自定义ContractResolver. 这样,您不必在Dictionary<K,V>每次序列化时进行子类化或应用转换,如其他答案中所建议的那样。

以下解析器将导致所有字典序列化为具有“Key”和“Value”属性的对象数组:

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);
    }
}

要使用解析器,请将其添加到您的JsonSerializerSettings,然后将设置传递给JsonConvert.SerializeObject()这样的:

JsonSerializerSettings settings = new JsonSerializerSettings();
settings.ContractResolver = new DictionaryAsArrayResolver();

string json = JsonConvert.SerializeObject(obj, settings);

这是一个工作演示

于 2014-07-31T16:48:37.683 回答
29

啊,事实证明这和我希望的一样简单。我Dictionary<k,v>的已经被子类化了,我发现我可以用[JsonArrayAttribute]. 这正是我需要的格式;

"MyDict": [
  {
    "Key": "Apples",
    "Value": {
        "Taste": 1341181398,
        "Title": "Granny Smith",
    }
  },
  {
    "Key:": "Oranges",
    "Value:": {
        "Taste": 9999999999,
        "Title": "Coxes Pippin",
    }
  },
]
于 2012-10-05T21:57:47.190 回答
14

对于这个例子,我将使用字典:

var myDict = new Dictionary<string,string>() { 
    {"a","123"}, 
    {"b","234"}, 
    {"c","345"} 
};

将(与Newtonsoft.Json.JsonConvert.SerializeObject(myDict))序列化为:

{"a":"123","b":"234","c":"345"}

您可以使用 LINQ 进行转换以创建匿名对象,并将其序列化:

 var myDictTransformed = from key in myDict.Keys
                         select new { k = key, v = myDict[key] };

或者你可以使用一个真实的对象

class MyDictEntry 
{
    public string k { get; set; }
    public string v { get; set; }
}

以及上述或替代 LINQ 语法:

var myDictTransformed = myDict.Keys.AsEnumerable()
                        .Select(key => new MyDictEntry{ 
                            k = key, 
                            v = myDict[key] 
                        });

无论哪种方式,这都会序列化为:

[
  {"k":"a", "v":"123"},
  {"k":"b", "v":"234"},
  {"k":"c", "v":"345"}
]

.NET 小提琴链接:https ://dotnetfiddle.net/LhisVW

于 2012-10-05T18:33:14.187 回答
12

我找到的最简单的解决方案是将您的转换Dictionary<string, string>List<KeyValuePair<string, string>>. JSON.NET 然后将您List转换为具有{ Key: 'keyname', Value: 'value' }. 如果您接受所需的模型更改并且不想继承您的Dictionary.

于 2015-08-25T01:09:26.560 回答
1

gregmac 的回答很有帮助,但不太奏效。以下是相同的想法......没有双关语。

var dictionaryTransformed = dictionary.Select(item => item.Key).Select(i => 
                        new {Key = i, Value = dictionary[i] });

或者当然

var dictionaryTransformed = dictionary.Select(item => 
                        new {item.Key, Value = dictionary[item.Key] });

然后到json

var json = (new JavaScriptSerializer()).Serialize( 
                        new { Container = dictionaryTransformed.ToArray() } )
于 2012-12-17T23:33:22.590 回答
0

我不确定为什么,但上面列出的 Brian Rogers 的自定义 ContractResolver 对我不起作用。它似乎在内部某个地方陷入了无限循环。可能是由于我的 json.net 设置的其他部分。

无论如何-这种解决方法对我有用。

public interface IStrongIdentifier
    {
        string StringValue { get; set; }
    }

public class StrongIdentifierKeyedDictionaryWrapper<TKey, TValue> : Dictionary<string, TValue>
        where TKey : IStrongIdentifier
    {
        public void Add(TKey key, TValue value)
        {
            base.Add(key.StringValue, value);
        }

        public void Remove(TKey key)
        {
            base.Remove(key.StringValue);
        }

        public TValue this[TKey index]
        {
            get => base[index.StringValue];
            set => base[index.StringValue] = value;
        }

        public bool ContainsKey(TKey key)
        {
            return base.ContainsKey(key.StringValue);
        }
    }
于 2018-11-20T21:05:40.340 回答