强制将空JSON数组转换为Dictionary类型

sti*_*til 3 c# json json.net

问题是,json_encode()PHP 中的函数为正在读取其输出的工具留下了歧义.在PHP中,列表和词典都是相同类型的array.

echo json_encode([]); // []
echo json_encode(["5" => "something"]); // {"5": "something"}
Run Code Online (Sandbox Code Playgroud)

在JSON.NET中,我想强制两者[]{"5": "something"}转换为Dictionary<string, string>类型.但它识别[]为Dictionary的禁止结构并抛出异常.

我可以快速保留空的JSON数组或强制它们转换为空的Dictionary类型吗?

最终解决方案

我修改了接受的答案,使其成为通用的,并可重用于其他类型.

public class DictionaryOrEmptyArrayConverter<T,F> : JsonConverter
{
    public override bool CanWrite { get { return false; } }
    public override bool CanConvert(Type objectType)
    {
        return objectType == typeof(Dictionary<T, F>);
    }
    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        switch (reader.TokenType)
        {
            case JsonToken.StartArray:
                reader.Read();
                if (reader.TokenType == JsonToken.EndArray)
                    return new Dictionary<T, F>();
                else
                    throw new JsonSerializationException("Non-empty JSON array does not make a valid Dictionary!");
            case JsonToken.Null:
                return null;
            case JsonToken.StartObject:
                var tw = new System.IO.StringWriter();
                var writer = new JsonTextWriter(tw);
                writer.WriteStartObject();
                int initialDepth = reader.Depth;
                while (reader.Read() && reader.Depth > initialDepth)
                {
                    writer.WriteToken(reader);
                }
                writer.WriteEndObject();
                writer.Flush();
                return JsonConvert.DeserializeObject<Dictionary<T, F>>(tw.ToString());
            default:
                throw new JsonSerializationException("Unexpected token!");
        }
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        throw new NotImplementedException();
    }
}
Run Code Online (Sandbox Code Playgroud)

您应该将它与JSON对象一起使用,如下例所示:

public class Company
{
    public string industry_name { get; set; }

    [JsonConverter(typeof(DictionaryOrEmptyArrayConverter<int, Upgrade>))]
    public Dictionary<int, Upgrade> upgrades { get; set; }
}

public class Upgrade
{
    public int level { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

它允许您使用DeserializeObject方法快速将JSON字符串转换为对象:

var result = JsonConvert.DeserializeObject<Company>(jsonString);
Run Code Online (Sandbox Code Playgroud)

小智 5

你可以使用自定义来完成它JsonConverter,虽然我认为这个解决方案有点不尽如人意:

    private class DictionaryConverter : JsonConverter
    {
        public override bool CanWrite { get { return false; } }
        public override bool CanConvert(Type objectType)
        {
            return objectType == typeof(Dictionary<string, string>);
        }
        public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
        {
            if (reader.TokenType == JsonToken.StartArray)
            {
                reader.Read();
                if (reader.TokenType == JsonToken.EndArray)
                    return new Dictionary<string, string>();
                else
                    throw new JsonSerializationException("Non-empty JSON array does not make a valid Dictionary!");
            }
            else if (reader.TokenType == JsonToken.Null)
            {
                return null;
            }
            else if (reader.TokenType == JsonToken.StartObject)
            {
                Dictionary<string, string> ret = new Dictionary<string, string>();
                reader.Read();
                while (reader.TokenType != JsonToken.EndObject)
                {
                    if (reader.TokenType != JsonToken.PropertyName)
                        throw new JsonSerializationException("Unexpected token!");
                    string key = (string)reader.Value;
                    reader.Read();
                    if (reader.TokenType != JsonToken.String)
                        throw new JsonSerializationException("Unexpected token!");
                    string value = (string)reader.Value;
                    ret.Add(key, value);
                    reader.Read();
                }
                return ret;
            }
            else
            {
                throw new JsonSerializationException("Unexpected token!");
            }
        }

        public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
        {
            throw new NotImplementedException();
        }
    }
Run Code Online (Sandbox Code Playgroud)

请参阅https://dotnetfiddle.net/zzlzH4上的此代码