2015-11-05 120 views
1

我有解析給定的JSON數據這個奇怪的問題。我有這樣的JSON結構:Newtonsoft JSON.NET解析到自定義鍵/值對對象的數組

{"value":[ 
    {"street":"Karlova 25"}, 
    {"city":"Prague"}, 
    {"gpsLat":"50.1571"}, 
    {"gpsLon":"15.0482"} 
]} 

如何使用Newtonsoft JSON.NET庫解析這個結構呢?我試圖用我自己的JsonConverter類:

public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer){ 
    JArray jarray = (JArray)((JTokenReader)reader).CurrentToken; 
    List<AddressValue> values = new List<AddressValue>(); 
    foreach (var jobj in jarray.Children<JObject>()){ 
    foreach (JProperty prop in jobj.Properties()){ 
     values.Add(new AddressValue() { Label = prop.Name, Value = prop.Value.ToString() }); 
    } 
    } 
    return values.ToArray(); 
} 

class AddressValue{ 
    public string Label { get; set; } 
    public string Value { get; set; } 
} 

,但我有一個例外:

Exception thrown: 'Newtonsoft.Json.JsonSerializationException' in Newtonsoft.Json.DLL 

Additional information: Unexpected token when deserializing object: StartObject. Path 'value[0]'. 

編輯: 我也試過這個保存到詞典:

[JsonProperty(PropertyName = "value")] 
public Dictionary<string, string> Value{get; set;} 

但我還有一個例外:

$exception {"Cannot deserialize the current JSON array (e.g. [1,2,3]) into type 'System.Collections.Generic.Dictionary`2[System.String,System.String]' because the type requires a JSON object (e.g. {\"name\":\"value\"}) to deserialize correctly.\r\nTo fix this error either change the JSON to a JSON object (e.g. {\"name\":\"value\"}) or change the deserialized type to an array or a type that implements a collection interface (e.g. ICollection, IList) like List<T> that can be deserialized from a JSON array. JsonArrayAttribute can also be added to the type to force it to deserialize from a JSON array.\r\nPath 'param.value'."} 

我做錯了什麼?謝謝您的回答。

+0

在Visual Studio中,您可以編輯 - >選擇性粘貼 - >將JSON粘貼爲類,然後DeserializeObject (stringData)。 – crashmstr

+0

這不是一個數組,它是一個最糟糕的詞典,但是您應該能夠聲明一個Address類,其中的屬性與JSON鍵相同,並直接進行反序列化。 @crashmstr的評論可能適用於較新版本的VS,但創建目標類也是一種選擇。 – Eris

+0

我忘了補充說,但保存在字典中的數據也不工作。我將編輯我的問題:-) –

回答

1

看來,你要代表你的JSON一個Dictionary<string, string>作爲對象的數組,其中每個嵌套的對象已經從字典一個鍵和值。您可以通過以下轉爐做到這一點:

public class DictionaryToDictionaryListConverter<TKey, TValue> : JsonConverter 
{ 
    [ThreadStatic] 
    static bool disabled; 

    // Disables the converter in a thread-safe manner. 
    bool Disabled { get { return disabled; } set { disabled = value; } } 

    public override bool CanWrite { get { return !Disabled; } } 

    public override bool CanRead { get { return !Disabled; } } 

    public override bool CanConvert(Type objectType) 
    { 
     if (Disabled) 
      return false; 
     return typeof(IDictionary<TKey, TValue>).IsAssignableFrom(objectType); 
    } 

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) 
    { 
     if (reader.TokenType == JsonToken.Null) 
      return null; 
     var token = JToken.Load(reader); 
     var dict = (IDictionary<TKey, TValue>)(existingValue as IDictionary<TKey, TValue> ?? serializer.ContractResolver.ResolveContract(objectType).DefaultCreator()); 
     if (token.Type == JTokenType.Array) 
     { 
      foreach (var item in token) 
       using (var subReader = item.CreateReader()) 
        serializer.Populate(subReader, dict); 
     } 
     else if (token.Type == JTokenType.Object) 
     { 
      using (var subReader = token.CreateReader()) 
       serializer.Populate(subReader, dict); 

     } 
     return dict; 
    } 

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer) 
    { 
     var dict = (IDictionary<TKey, TValue>)value; 
     // Prevent infinite recursion of converters 
     using (new PushValue<bool>(true,() => Disabled, val => Disabled = val)) 
     { 
      serializer.Serialize(writer, dict.Select(p => new[] { p }.ToDictionary(p2 => p2.Key, p2 => p2.Value))); 
     } 
    } 
} 

public struct PushValue<T> : IDisposable 
{ 
    Action<T> setValue; 
    T oldValue; 

    public PushValue(T value, Func<T> getValue, Action<T> setValue) 
    { 
     if (getValue == null || setValue == null) 
      throw new ArgumentNullException(); 
     this.setValue = setValue; 
     this.oldValue = getValue(); 
     setValue(value); 
    } 

    // By using a disposable struct we avoid the overhead of allocating and freeing an instance of a finalizable class. 
    public void Dispose() 
    { 
     if (setValue != null) 
      setValue(oldValue); 
    } 
} 

然後按如下方式使用它在你的容器類:

public class RootObject 
{ 
    [JsonProperty("value")] 
    [JsonConverter(typeof(DictionaryToDictionaryListConverter<string, string>))] 
    public Dictionary<string, string> Value { get; set; } 
} 

注意,在讀取時轉換器將拋出一個異常,如果該鍵不唯一。

更新

對於AddressValue,你可以使用下面的轉換器:

public class AddressValueConverter : JsonConverter 
{ 
    public override bool CanConvert(Type objectType) 
    { 
     return objectType == typeof(AddressValue); 
    } 

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) 
    { 
     if (reader.TokenType == JsonToken.Null) 
      return null; 
     var addressValue = (existingValue as AddressValue ?? new AddressValue()); 
     var token = JObject.Load(reader); 
     var property = token.Properties().SingleOrDefault(); 
     if (property != null) 
     { 
      addressValue.Label = property.Name; 
      addressValue.Value = (string)property.Value; 
     } 
     return addressValue; 
    } 

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer) 
    { 
     var addressValue = (AddressValue)value; 
     serializer.Serialize(writer, new Dictionary<string, string> { { addressValue.Label, addressValue.Value } }); 
    } 
} 

然後按如下方式使用它:

[JsonConverter(typeof(AddressValueConverter))] 
public class AddressValue 
{ 
    public string Label { get; set; } 
    public string Value { get; set; } 
} 

public class RootObject 
{ 
    [JsonProperty("value")] 
    public List<AddressValue> Value { get; set; } 
} 
+0

謝謝!它完美的工作! –

3

你不需要重新發明輪子。這種功能性已經在起作用。 像下面創建類:

public class Value 
{ 
    public string street { get; set; } 
    public string city { get; set; } 
    public string gpsLat { get; set; } 
    public string gpsLon { get; set; } 
} 

public class MyClass 
{ 
    public List<Value> value { get; set; } 
} 

現在,你可以簡單地反序列化JSON你到你的POCO對象。

MyClass result = JsonConvert.DeserializeObject<MyClass>(youJson); 
+0

謝謝,但這不是我所需要的。我從Web服務中獲得這個JSON,將來可能會更改JSON。我需要雙方(密鑰和值)添加動態,因爲它們來自服務器。 –