我想從Web響應中反序列化JSON。這是一個典型的迴應:如何使用自定義的JsonConverter來僅對Json.NET中的子對象進行反序列化?
{
"response": {
"garbage": 0,
"details": [
{
"id": "123456789"
}
]
}
}
但是,這種格式是不可取的。理想情況下,他們的迴應是剛剛
{
"id": "123456789"
}
,以便它可以deserialised爲對象像
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()
正在對同一對象重複使用,它從來沒有得到deserialised。我認爲這是因爲我通過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'的第二種方法。此外,如果您不使用它,則不必聲明「垃圾」 - 只有直接的父母纔會這樣做。 – aaron
有了很多嵌套的更大的JSON文件並不像代碼生成器那樣煩人。然而,創建一堆虛擬父類似乎並不優雅。 – Mark
@aaron有2天的等待期。 – Mark