2015-12-21 107 views
2

我有以下JSON和我使用Json.NET(Newtonsoft.Json):C#解析JSON問題

{ 
    "total_items": "62", 
    "page_number": "6", 
    "page_size": "10", 
    "page_count": "7", 
    "cars": { 
    "car": [  
     { 
     "car_name": "Honda", 
     "engines": { 
      "engine": [ <-- HONDA has multiple engines, so this is an array 
      { 
       "name": "1.2L" 
      }, 
      { 
       "name": "1.8L" 
      } 
      ] 
     }, 
     "country": "Japan" 
     "image": { 
      "thumb": { 
       "url": "http://image_path/Honda.jpg" <-- Image provided 
      } 
     } 
     }, 
     { 
     "car_name": "Ford", 
     "engines": { 
      "engine": { <-- FORD has single engine, so this is an object 
       "name": "2.2L" 
      } 
     }, 
     "country": "Japan" 
     "image": null <-- image is null 
     }, 
     { 
     "car_name": "VW", 
     "engines": null, <-- VW has no engines, so this is null 
     "country": "Germany" 
     "image": null <-- image is null 
     } 
    ] 
    } 
} 

而且我有以下Car對象:

class Car 
{ 
    public Car() { } 

    public string Name { get; set; } 
    public string Country { get; set; } 
    public List<String> EngineNames { get; set; } 
} 

我需要處理所有以上3種情況(HONDA陣列,FORD對象陣列,VW空白陣列)。如果它不爲空,則獲取所有引擎名稱。因此,例如上面,我EngineNames列表中3輛汽車將是:

Honda.EngineNames = {"1.2L", "1.8L"} // array in JSON 
Ford.EngineNames = {"2.2L"} //object in JSON 
VW.EngineNames = null //null in JSON 

我需要解析上面的JSON得到車資料。我解析car_name和國家,但我不知道如何通過處理上述3種情況來解析所有引擎名稱。

private Cars GetCars(string json) 
{ 
    dynamic data = (JObject)JsonConvert.DeserializeObject(json); 

    foreach (dynamic d in data.cars.car) 
    { 
     Car c = new Car(); 

     c.Name = (string)d.SelectToken("car_name"); 
     c.Country = (string)d.SelectToken("country"); 

     // PROBLEM: This works fine for array or null in JSON above (HONDA and VW), but it errors on JSON object (in case of FORD) 
     // When handling FORD, I get error "'Newtonsoft.Json.Linq.JProperty' does not contain a definition for 'name'" 
     c.EngineNames = (d.engines != null ? ((IEnumerable)d.engines.engine).Cast<dynamic>().Select(e => (string)e.name) : null); 

     CarList.Add(c); 
    } 
    return CarList; 
} 
+0

爲什麼不將單個項目引擎放入一個項目的數組中?像:''引擎「:[{」name「:」2.2L「}]' – Steve

+0

@Steve JSON提供給我,我沒有做到。這是怎麼回事,我需要解析它。謝謝 – pixel

+4

可能的重複[如何使用JSON.net處理同一個屬性的單個項目和數組](http://stackoverflow.com/questions/18994685/how-to-handle-both-a-single- item-and-an-array-for-the-same-property-using-json-n) – Rob

回答

2

使用從here轉換器(最初提出的重複,但這個問題曾與JSON其他一些問題)

你的類結構有待修改了一下。

望着這JSON:

"cars": { <-- cars is an object, not an array 
    "car": [ <-- the cars object actually contains the array 
     { 
     "car_name": "Honda", 
     "engines": { <-- same goes for this 
      "engine": [ 
      { 

因此,你需要寫包裝類,以正確反映了JSON。下面是我想出什麼用:

public class Root 
{ 
    public CarHolder Cars {get;set;} 
} 
public class CarHolder 
{ 
    public IList<Car> Car { get; set; } 
} 
public class Car 
{ 
    public Car() { } 

    public string car_name { get; set; } 
    public string Country { get; set; } 

    public EngineHolder Engines { get; set; } 
} 
public class EngineHolder 
{ 
    [JsonConverter(typeof(SingleOrArrayConverter<Engine>))] 
    public List<Engine> Engine { get; set; } 
} 
public class Engine 
{ 
    public string Name { get; set; } 
} 

而且使用轉換從上面的問題:

public class SingleOrArrayConverter<T> : JsonConverter 
{ 
    public override bool CanConvert(Type objectType) 
    { 
     return (objectType == typeof(List<T>)); 
    } 

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) 
    { 
     JToken token = JToken.Load(reader); 
     if (token.Type == JTokenType.Array) 
     { 
      return token.ToObject<List<T>>(); 
     } 
     return new List<T> { token.ToObject<T>() }; 
    } 

    public override bool CanWrite 
    { 
     get { return false; } 
    } 

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer) 
    { 
     throw new NotImplementedException(); 
    } 
} 

用法:

var result = JsonConvert.DeserializeObject<Root>(jsonStr); 
Console.WriteLine(result.Cars.Car[0].Engines.Engine[0].Name == "1.2L"); 
Console.WriteLine(result.Cars.Car[0].Engines.Engine[1].Name == "1.8L"); 
Console.WriteLine(result.Cars.Car[1].Engines.Engine[0].Name == "2.2L"); 
Console.WriteLine(result.Cars.Car[2].Engines == null); 

所有打印true

通過循環汽車&發動機

foreach(var car in result.Cars.Car) 
{ 
    if (car.Engines != null) 
    { 
     foreach(var engine in car.Engines.Engine) 
     { 
      var engineName = engine.Name; 
     } 
    } 
} 
+0

是的,我得到了相同的結果,但是我有問題想知道如何循環顯示每個車的「結果」var和print car_name,country以及所有引擎?非常感謝你 – pixel

+1

@ dbnex14我已經添加了一個示例來循環顯示結果 – Rob

+0

感謝Rob。我還有一個問題,我更新了我的原始帖子,我在JSON中添加了「圖片」。我需要獲取圖像url,但處理圖像可能爲空的情況。如果您想要我,我可以發佈新問題並將其標記爲已回答。非常感謝 – pixel

1

你應該可以使用它作爲你的類結構;

public class Rootobject 
{ 
    public string total_items { get; set; } 
    public string page_number { get; set; } 
    public string page_size { get; set; } 
    public string page_count { get; set; } 
    public Cars cars { get; set; } 
} 

public class Cars 
{ 
    public Car[] car { get; set; } 
} 

public class Car 
{ 
    public string car_name { get; set; } 
    public Engines engines { get; set; } 
    public string country { get; set; } 
} 

public class Engines 
{ 
    public object engine { get; set; } 
} 

//I created below class manually 
public class Engine 
{ 
    public string name { get; set; } 
} 

我用VS的內置功能來生成這個。腳步;

  1. 打開一個新的cs文件。
  2. 複製您的JSON
  3. 轉到編輯菜單>粘貼特殊
  4. 選擇粘貼JSON作爲類

一旦做到這一點,它應該只是一個創建了兩個方法,序列化的問題,並反序列化。

與還原序列化/ deserialise方法

 private static T Deserialise<T>(string json) 
     { 
      var myopject = JsonConvert.DeserializeObject<T>(json); 
      return myopject; 
     } 

     private static string Serialise<T>(T value) 
     { 
      var mycontent = JsonConvert.SerializeObject(value); 
      return mycontent; 
     } 

我們測試上面的方法更新,你可以做到這一點。

  var jsonstring = @"{ 
       ""total_items"": ""62"", 
       ""page_number"": ""6"", 
       ""page_size"": ""10"", 
       ""page_count"": ""7"", 
       ""cars"": { 
       ""car"": [ 
        { 
        ""car_name"": ""Honda"", 
        ""engines"": { 
         ""engine"": [ 
         { 
          ""name"": ""1.2L"" 
         }, 
         { 
          ""name"": ""1.8L"" 
         } 
         ] 
        }, 
        ""country"": ""Japan"" 
        }, 
        { 
        ""car_name"": ""Ford"", 
        ""engines"": { 
         ""engine"": { 
         ""name"": ""2.2L"" 
         } 
        }, 
        ""country"": ""Japan"" 
        }, 
        { 
        ""car_name"": ""VW"", 
        ""engines"": null, 
        ""country"": ""Germany"" 
        } 
       ] 
       } 
      }"; 
      var myobject = Deserialise<Rootobject>(jsonstring); 

      //if you want to parse engines you can do something like this. 

      if (myobject.cars != null && myobject.cars.car != null && myobject.cars.car.Any()) 
     { 
      foreach (Car car in myobject.cars.car) 
      { 
       if (car.engines != null && car.engines.engine != null) 
       { 
        bool isList = false; 
        try 
        { 
         var eng = Deserialise<Engine>(car.engines.engine.ToString()); 
        } 
        catch 
        { 
         isList = true; 
        } 
        if (isList) 
        { 
         try 
         { 
          var eng = Deserialise<List<Engine>>(car.engines.engine.ToString()); 
         } 
         catch 
         { 
          Debug.WriteLine("Not a list"); 
         } 
        } 
       } 
      } 
     } 

      var myjson = Serialise(myobject); 
+0

謝謝。我將如何反序列化這個創建的類。你是否介意提供示例,因爲我是新手。謝謝, – pixel

+1

回答更新。 –

+0

我得到的錯誤「預期的類,委託,枚舉,接口或結構」和T在私有靜態T Deserialise (字符串json)是紅色下劃線 – pixel