我希望能够从Web响应中反序列化JSON。以下是一个典型的响应:
{
"response": {
"garbage": 0,
"details": [
{
"id": "123456789"
}
]
}
}
然而,这种格式并不理想。理想情况下,响应应该只包含:
{
"id": "123456789"
}
这样它就可以被反序列化成一个对象,如下所示:
public class Details {
[JsonProperty("id")]
public ulong Id { get; set; }
}
由于服务器不受我控制(它是一个公共API),所以我希望修改反序列化过程以实现我想要的格式。
我试图利用自定义的JsonConverter
来实现这一目标。 思路是跳过令牌,直到找到Details
的期望起始点进行反序列化。 但是,我不确定应该在反序列化过程中的哪个位置使用它。
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
using Newtonsoft.Json;
namespace ConsoleApp2 {
class Program {
static void Main(string[] args) {
// Simulating a stream from WebResponse.
const string json = "{"response":{"garbage":0,"details":[{"id":"123456789"}]}}";
byte[] bytes = System.Text.Encoding.UTF8.GetBytes(json);
Stream stream = new MemoryStream(bytes);
Details details = Deserialise<Details>(stream);
Console.WriteLine($"ID: {details.Id}");
Console.Read();
}
public static T Deserialise<T>(Stream stream) where T : class {
using (StreamReader reader = new StreamReader(stream, Encoding.UTF8)) {
JsonSerializerSettings settings = new JsonSerializerSettings {
MissingMemberHandling = MissingMemberHandling.Ignore,
DateParseHandling = DateParseHandling.None
};
settings.Converters.Add(new DetailConverter());
return JsonSerializer.Create(settings).Deserialize(reader, typeof(T)) as T;
}
}
}
public class Details {
[JsonProperty("id")]
public ulong Id { get; set; }
}
public class DetailConverter : JsonConverter {
public override void WriteJson(JsonWriter writer,
object value,
JsonSerializer serializer) {
throw new NotImplementedException();
}
public override object ReadJson(JsonReader reader,
Type objectType,
object existingValue,
JsonSerializer serializer) {
if (reader.Depth == 0) {
while (!(reader.Path.Equals("response.details[0]", StringComparison.OrdinalIgnoreCase) && reader.TokenType == JsonToken.StartObject)) {
reader.Read();
}
try {
return serializer.Deserialize(reader, objectType);
} finally {
reader.Read(); // EndArray - details
reader.Read(); // EndObject - response
reader.Read(); // EndObject - root
}
}
return serializer.Deserialize(reader, objectType);
}
public override bool CanWrite => false;
public override bool CanConvert(Type objectType) => true;
}
}
目前的情况是,由于重复使用
DetailConverter.ReadJson()
并且它从未被反序列化,导致堆栈溢出。我认为问题出在我通过JsonSerializerSettings
将DetailConverter
设置为“全局”转换器。我认为问题在于何时以及如何使用我的转换器,而不是其内部工作方式。
我已经让类似的
DetailConverter
适用于以下结构。但是,虽然来自details
的数组已被删除,但由于嵌套和未使用的属性,这仍然是不可取的。public class Root {
[JsonProperty("response")]
public Response Response { get; set; }
}
public class Response {
[JsonProperty("details")]
[JsonConverter(typeof(DetailConverter))]
public Details Details { get; set; }
[JsonProperty("garbage")]
public uint Garbage { get; set; }
}
public class Details {
[JsonProperty("id")]
public ulong Id { get; set; }
}
我以为将转换器扩展到整个JSON而不仅仅是一个属性会很简单。我错在哪里了?
JsonConverter
来采用第二种方法。此外,如果你不使用"Garbage",就不必声明它——只需直接使用父级即可。 - aaron