5

服务器返回一个 JSON 字符串值,它是一个 URL 查询字符串:

{
    "parameters": "key1=value1&key2=value2"
}

我设置了一个属性来接收它,并将其转换为 aDictionary作为反序列化过程的一部分:

具有属性的JsonConverter属性:

[JsonConverter(typeof(QueryStringToDictionaryJsonConverter))]
public Dictionary<string, string> Parameters { get; set; }

转换器:

public class QueryStringToDictionaryJsonConverter : JsonConverter<Dictionary<string, string>> {

    public override Dictionary<string, string> Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) {

        var queryString = reader.GetString();
        if (string.IsNullOrEmpty(queryString)) return null;

        return QueryHelpers.ParseQuery(queryString).ToDictionary(e => e.Key, e => string.Join(",", e.Value.ToArray()));

    }

    ...
}

这应该有效。

但它甚至没有到达我的转换器。

据我所知,JsonSerializer.DeserializeAsync<T>(myJson)看到属性的类型是 a Dictionary,因此它尝试自行解析值,但失败(结果异常是“无效转换”,因为它试图GetEnumerable()等等)。我的转换器中的断点甚至从未被击中。

我可以通过将属性设置为 anobject然后将其转换为Dictionary稍后使用它的位置来使其工作,但这是一个丑陋的解决方案。

有没有办法强制JsonSerializer.DeserializeAsync<T>(myJson)只使用我的转换器,而不是让它自己变得聪明?

(我在 .NET Core 3 中使用 Microsoft 的 System.Text.Json)

4

2 回答 2

2

好的,所以这可能是System.Text.Json.

这是我目前为其他需要解决方案的人使用的解决方法。

首先,我使用[JsonPropertyName]and设置了两个反序列化属性[JsonIgnore]

[JsonPropertyName("parameters"), JsonConverter(typeof(QueryStringToDictionaryJsonConverter))]
public object ParametersObject { get; set; }

[JsonIgnore]
public Dictionary<string, string> Parameters => ParametersObject as Dictionary<string, string>;

然后在 中JsonConverter,我允许object作为类型:

public override bool CanConvert(Type typeToConvert) {
    if (typeToConvert == typeof(object)) return true;
    return base.CanConvert(typeToConvert);
}

我的反序列化类的消费者只使用该Parameters属性,如果修复了此错误并且我将类更改回我想要的方式,它将继续正常工作。

于 2019-10-03T11:10:36.140 回答
0

我将创建一个包装器并为该包装器创建一个转换器。

[JsonConverter( typeof( QueryStringDictionaryConverter ) )]
class QueryStringDictionary : Dictionary<string,string> { }

class QueryStringDictionaryConverter : JsonConverter<QueryStringDictionary>
{
    ... 
}

class MyClass
{
    public QueryStringDictionary Parameters { get; set; }
}

或者你可以使用JsonSerializerOptions

class MyOtherClass
{
   public Dictionary<string,string> Parameters { get; set; }
}
MyOtherClass Deserialize( string json )
{
    var options = new JsonSerializerOptions
    {
        Converters = { new QueryStringToDictionaryJsonConverter() }
    };
    return JsonSerializer.Deserialize<MyOtherClass>( json, options );  
} 

这种方法的一个潜在问题是转换器将用于所有Dictionary<string,string>属性,这可能不是预期的。对于原始问题中的简单示例,它可以正常工作。

于 2019-10-03T12:56:08.000 回答