【问题标题】:Json.NET serializing Dictionary<TKey, TValue> vs KeyValuePair<TKey, TValue>Json.NET 序列化 Dictionary<TKey, TValue> 与 KeyValuePair<TKey, TValue>
【发布时间】:2021-10-21 04:19:42
【问题描述】:

考虑下面的代码

var dict = new Dictionary<string, object>
{
    { "key 1", "value 1" },
    { "key 2", 123 }
};
var dictJson = JsonConvert.SerializeObject(dict); // yields {"key 1":"value 1","key 2":123}

var keyValuePair = dict.FirstOrDefault();
var keyValuePairJson = JsonConvert.SerializeObject(keyValuePair); // yields {"Key":"key 1","Value":"value 1"}

第一个问题是,为什么字典的第一个元素KeyValuePair&lt;TKey, TValue&gt;的json和Dictionary&lt;TKey, TValue&gt;的json不同?

第二个问题是,我怎样才能实现与序列化字典类似的 json,但只有一个项目而不是扩展集合?我的目标是有一个类似于下面的类,但没有将KeyValue 作为序列化json 中的属性。

public class Foo
{
    public KeyValuePair<string, object> Pair { get; set; }
}

【问题讨论】:

  • 你为什么期望它们是一样的? KeyValuePairDictionary 是完全不同的东西。
  • @Sweeper 获取字典的第一个元素,我希望序列化的 json 看起来相同,但只有一个元素。
  • Dictionary 可能有一个内置序列化程序,但KeyValuePair 恢复为默认的Object 序列化程序。
  • @IllusiveBrian 正确,但我相信它反过来。有一个 KeyValuePairConverter 可能会自动用于字典。但是当你输入一个 KeyValuePair 时,这是一件不常见的事情,它并没有被使用。因此,使用 SerializeObject 重载可以声明要使用的转换器应该会产生相同的结果。

标签: c# json .net dictionary json.net


【解决方案1】:

为什么字典第一个元素KeyValuePair&lt;TKey, TValue&gt;的json和Dictionary&lt;TKey, TValue&gt;的json不同?

字典和键值对是非常不同的东西。字典不是只是键值对的集合。这是字典的一种观点,当然,但说字典就是这样是非常错误的。

更具体地说,NewtonsoftJson 中有一个 JsonConverter 专门将 KeyValuePair 转换为 JSON 格式:

{ "Key": ..., "Value": ... }

KeyValuePairConverter.

请注意,此表单将键和值都转换为 JSON,这很可能是您将“键值”转换为 JSON 时想要的。将此与字典转换器在字典的键不是字符串时所做的比较 - 它只是调用 ToString 使其成为字符串:(

如何实现与序列化字典类似的 json,但只有一个项目而不是扩展集合?

你可以这样写JsonConverter

public class KeyValuePairObjectConverter : JsonConverter
{
    public override void WriteJson(JsonWriter writer, object? value, JsonSerializer serializer)
    {
        JToken t = JToken.FromObject(value);

        if (t.Type != JTokenType.Object)
        {
            t.WriteTo(writer);
        }
        else
        {
            JObject o = (JObject)t;
            string key = o.Value<string>("Key");
            var val = o["Value"];
            writer.WriteStartObject();
            writer.WritePropertyName(key);
            val.WriteTo(writer);
            writer.WriteEndObject();
        }
    }

    public override object? ReadJson(JsonReader reader, Type objectType, object? existingValue, JsonSerializer serializer)
        => throw new NotImplementedException();

    public override bool CanRead => false;
    public override bool CanConvert(Type t)
    {
        if (t.IsValueType && t.IsGenericType)
        {
            return (t.GetGenericTypeDefinition() == typeof(KeyValuePair<,>)) &&
                    t.GetGenericArguments()[0] == typeof(string);
        }

        return false;
    }
}

用法:

public class Foo
{
    [JsonConverter(typeof(KeyValuePairObjectConverter))]
    public KeyValuePair<string, object> Pair { get; set; }
}

【讨论】:

  • 太棒了,正是我想要的。我唯一能找到的一个问题是,当我在JsonConvert.DefaultSettings 中添加这个转换器时,它会抛出StackOverflowException。它卡在JToken t = JToken.FromObject(value); 线上。在这种情况下,对象的创建似乎需要手动处理,在这个答案stackoverflow.com/questions/22407321/… 中有解释
猜你喜欢
  • 1970-01-01
  • 2011-01-16
  • 2014-02-19
  • 2011-04-11
  • 2016-06-14
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多