使用.NET 4.0任务模式使用HTTPClient .ReadAsAsync将JSON反序列化为数组或列表


77

我正在尝试反序列化http://api.usa.gov/jobs/search.json?query=nursing+jobs使用.NET 4.0任务模式返回的JSON 。它将返回此JSON(“加载JSON数据” @ http://jsonviewer.stack.hu/)。

[
  {
    "id": "usajobs:353400300",
    "position_title": "Nurse",
    "organization_name": "Indian Health Service",
    "rate_interval_code": "PA",
    "minimum": 42492,
    "maximum": 61171,
    "start_date": "2013-10-01",
    "end_date": "2014-09-30",
    "locations": [
      "Gallup, NM"
    ],
    "url": "https://www.usajobs.gov/GetJob/ViewDetails/353400300"
  },
  {
    "id": "usajobs:359509200",
    "position_title": "Nurse",
    "organization_name": "Indian Health Service",
    "rate_interval_code": "PA",
    "minimum": 42913,
    "maximum": 61775,
    "start_date": "2014-01-16",
    "end_date": "2014-12-31",
    "locations": [
      "Gallup, NM"
    ],
    "url": "https://www.usajobs.gov/GetJob/ViewDetails/359509200"
  },
  ...
]

索引动作:

  public class HomeController : Controller
  {
    public ActionResult Index()
    {
      Jobs model = null;
      var client = new HttpClient();
      var task = client.GetAsync("http://api.usa.gov/jobs/search.json?query=nursing+jobs")
        .ContinueWith((taskwithresponse) =>
        {
          var response = taskwithresponse.Result;
          var jsonTask = response.Content.ReadAsAsync<Jobs>();
          jsonTask.Wait();
          model = jsonTask.Result;
        });
      task.Wait();
      ...
     }

职位和职位类别:

  [JsonArray]
  public class Jobs { public List<Job> JSON; }

  public class Job
  {
    [JsonProperty("organization_name")]
    public string Organization { get; set; }
    [JsonProperty("position_title")]
    public string Title { get; set; }
  }

当我设置断点jsonTask.Wait();并检查jsonTask状态为“故障”时。InnerException是“ Type ProjectName.Jobs不是集合”。

我从没有JsonArray属性的Jobs类型开始,而Jobs作为数组(Job [])开始出现此错误。

  public class Jobs { public Job[] JSON; }

    +       InnerException  {"Cannot deserialize the current JSON array (e.g. [1,2,3]) into type 'ProjectName.Models.Jobs' because the type requires a JSON object (e.g. {\"name\":\"value\"}) to deserialize correctly.\r\n
    To 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\n
Path '', line 1, position 1."}  System.Exception {Newtonsoft.Json.JsonSerializationException}

如何使用.NET 4.0任务模式处理此站点的JSON?在进入await async.NET 4.5中的模式之前,我想使此工作正常。

答案更新:

这是一个使用带有brumScouse答案的.NET 4.5异步等待模式的示例。

 public async Task<ActionResult>Index()
 {
    List<Job> model = null;
    var client = newHttpClient();

    // .NET 4.5 async await pattern
    var task = await client.GetAsync(http://api.usa.gov/jobs/search.json?query=nursing+jobs);
    var jsonString = await task.Content.ReadAsStringAsync();
    model = JsonConvert.DeserializeObject<List<Job>>(jsonString);
    returnView(model);
 }

您将需要引入System.Threading.Tasks名称空间。
注意:没有.ReadAsString可用的方法,.Content这就是为什么我使用该.ReadAsStringAsync方法。


2
你试过了ReadAsAsync<Job[]>()吗?
2014年

1
不起作用,在taskwithresponse的.Result上产生此错误。错误1'System.Threading.Tasks.Task'不包含'Result'的定义,并且找不到扩展方法'Result'接受类型为'System.Threading.Tasks.Task'的第一个参数(您是否缺少使用指令还是程序集引用?)
Joe

这没有任何意义,更改in的类型ReadAsAsync()无法更改代码的行为。
svick 2014年

好吧。它嵌入在ContinueWith语句中。请创建一个新的MVC4应用程序(在VS 2012中工作)并粘贴到该控制器和两个类中吗?您可以复制该错误吗?如果是这样,那么您是否能够提出解决问题的经过测试的解决方案?
2014年

Answers:


98

可以使用Json2csharp.com网站之类的东西来代替手动建模。粘贴在示例JSON响应中,越饱满越好,然后提取生成的生成类。至少,这消除了一些活动部分,将使您获得csharp中JSON的形状,从而使序列化程序更轻松,而且您不必添加属性。

使其工作,然后对您的类名进行修改,以符合您的命名约定,并在以后添加属性。

编辑:好吧,经过一番混乱,我已经成功地将结果反序列化为作业列表(我使用Json2csharp.com为我创建了类)

public class Job
{
        public string id { get; set; }
        public string position_title { get; set; }
        public string organization_name { get; set; }
        public string rate_interval_code { get; set; }
        public int minimum { get; set; }
        public int maximum { get; set; }
        public string start_date { get; set; }
        public string end_date { get; set; }
        public List<string> locations { get; set; }
        public string url { get; set; }
}

然后对您的代码进行编辑:

        List<Job> model = null;
        var client = new HttpClient();
        var task = client.GetAsync("http://api.usa.gov/jobs/search.json?query=nursing+jobs")
          .ContinueWith((taskwithresponse) =>
          {
              var response = taskwithresponse.Result;
              var jsonString = response.Content.ReadAsStringAsync();
              jsonString.Wait();
              model = JsonConvert.DeserializeObject<List<Job>>(jsonString.Result);

          });
        task.Wait();

这意味着您可以摆脱包含的对象。值得注意的是,这不是与任务相关的问题,而是反序列化的问题。

编辑2:

有一种方法可以获取JSON对象并在Visual Studio中生成类。只需复制选择的JSON,然后编辑>选择性粘贴>将JSON粘贴为类。整个页面专门用于此:

http://blog.codeinside.eu/2014/09/08/Visual-Studio-2013-Paste-Special-JSON-And-Xml/


我使用Fiddler来发现JSON,它表明返回了多个{}。Json2csharp.com是一个不错的工具,但是它没有显示出多个{}。我已经成功将JsonProperty属性与其他URL结合使用,所以我的问题不在于JsonProperty属性,而是与如何在.NET 4.0中使用Task模式将返回的JSON反序列化为数组或列表。
2014年

是的,正如我从一开始就说的那样,这是一个JSON反序列化问题。捕获字符串,然后通过JsonConverting做到了这一点,现在可以了。无需修改Job类型,我保留了具有JsonProperty属性的原始Job类型。
2014年

9
如果您只是要同步等待它们完成,那么就不要使用异步方法来获取数据了。如果您只是要同步等待,则最好从一开始就使用同步方法。如果要使用异步方法,则代码实际上应该是异步的。
Servy 2014年

2
@Joe在什么方面?他注意到有关可以修复的代码有些奇怪,请弄清楚。异步代码变得混乱,Servy很好。也许评论更适合,但无论哪种方式,下注不必是永久的。
Nate-Wilkins 2014年

1
那个第二次编辑让我很开心。谢谢!
smm

19
var response = taskwithresponse.Result;
          var jsonString = response.ReadAsAsync<List<Job>>().Result;

17
对于任何想知道扩展方法在哪里的人:Microsoft.AspNet.WebApi.Client nuget包。
Jimmyt1988 '16

5
永远不要使用,.Result因为它会阻塞线程。应该使用var jsonString = await taskwithresponse.ReadAsAsync<List<Job>>()
Dave Black

5

返回类型取决于服务器,有时响应的确是JSON数组,但以文本/纯文本形式发送

在请求中设置接受标头应获得正确的类型:

client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json")); 

然后可以将其序列化为JSON列表或数组。感谢@svick的评论,这使我很好奇它应该起作用。

我未配置接受标头的异常是System.Net.Http.UnsupportedMediaTypeException。

以下代码更简洁并且可以正常工作(未经测试,但对我而言有效):

    var client = new HttpClient();
    client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
    var response = await client.GetAsync("http://api.usa.gov/jobs/search.json?query=nursing+jobs");
    var model = await response.Content.ReadAsAsync<List<Job>>();

您忘记等待到的呼叫ReadAsAsync。最后一行应为:var model = await response.Content.ReadAsAsync<List<Job>>();
Dave Black
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.