Web Api 2使用集合属性反序列化LINQ2SQL对象

时间:2015-01-28 05:43:23

标签: ajax json serialization asp.net-web-api2

上下文很长,所以我将从问题开始:为什么InductionQAs集合属性没有被重新水合?

在Web API 2方法中,LINQ生成一个具有集合属性ic的对象InductionQAs。在我消除了一个巨大的无关对象图并阻止循环引用之后,该方法返回ic。这是代码。为简洁起见,已删除错误处理。

// GET: api/InductionContent/5
[Authorize]
public object Get(int id)
{
  var dc = new Models.InductionsDataContext();
  var user = GetUser(dc);
  var ic = dc.InductionContents.Single(c => c.InductionContentID == id);
  ic.User.InductionContents.Clear(); //don't ship this it can be huge
  return ic;
}

这里是调用JS代码。忽略授权标题业务,一切正常。

var token, headers;
$.ajax({ //get session token
  url: "api/token",
  headers: { Authorization: "Basic " + btoa("username:password") },
}).done(function (result) {
  localStorage.setItem('access_token', token = result.access_token);
  var headers = { Authorization: "Session " + token };
  $.ajax({ //get an induction content object 
    url: "api/InductionContent/5",
    headers: headers
  }).done(function (ic) {
    //at this point we have an object graph
    var json = JSON.stringify(ic);
  });
});

这是JSON.stringify(ic)返回的内容:

{
    "InductionContentID": 5,
    "InductionTemplateID": 1,
    "InductionContent1": "<p>We recognise that it is ...(redacted)</p>",
    "InductionContentOrder": 301,
    "Caption": "Workplace Health and Safety",
    "OwnerId": 0,
    "ParentId": 0,
    "TopicId": 5,
    "InductionQAs": [
        {
            "InductionQAID": 1,
            "InductionContentID": 5,
            "Question": "Who is responsible for ...(redacted)",
            "Answers": "A|B|C|*D",
            "InductionQAOrder": 1
        }
    ],
    "User": null
}

一切都很好。现在我们往返它。

$.ajax({
  type: "post",
  url: "api/InductionContent",
  headers: headers,
  data: ic
});

这会调用以下Web方法:

// POST: api/InductionContent
[Authorize]
public void Post([FromBody]Models.InductionContent ic)
{
  //redacted
}

调用该方法,ic有一个值。 ic.User包含数据,但对ic.InductionQAs的检查表明它尚未初始化。

序列化配置如下:

protected void Application_Start()
{
  GlobalConfiguration.Configure(WebApiConfig.Register);
  GlobalConfiguration.Configuration.Formatters.JsonFormatter.SerializerSettings
    .ReferenceLoopHandling = Newtonsoft.Json.ReferenceLoopHandling.Ignore;
  GlobalConfiguration.Configuration.MessageHandlers
    .Add(new AuthenticationHandler(CreateConfiguration()));
  FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
  RouteConfig.RegisterRoutes(RouteTable.Routes);
  BundleConfig.RegisterBundles(BundleTable.Bundles);
}

1 个答案:

答案 0 :(得分:0)

MVC很聪明但不够聪明。它需要一个提示。 &#34;往返&#34; call 应该看起来像这样:

    $.ajax({
      type: "post",
      url: "api/InductionContent",
      contentType: "text/json",
      headers: headers,
      data: JSON.stringify(ic)
    }).done(function (result) {
      //etc
    });

这里有两点需要注意。

  • 我们假设手动控制有效载荷编码
  • 我们使用编码
  • 注释了帖子

另一件事......

完成所有工作之后我决定该方法应该返回刚刚建立的数据库行的id。将方法返回类型更改为int并附加最后一行

return nc.InductionContentId;

应该已经足够了,但是ajax调用的done方法完全没有触发。对Fiddler的调查显示,id值确实已经返回,内容类型为text / json

为jQuery ajax调用指定内容类型时,响应使用相同的内容类型。如果返回值是一个对象,这将正常工作。该对象将被序列化为JSON,另一端将被解析以提供结果对象。

但对于像int这样的简单值,这在解析阶段就会被解除。要解决此问题,请使用dataType属性返回纯文本的内容类型。

$.ajax({
  type: "post",
  url: "api/InductionContent",
  contentType: "text/json",
  dataType: "text",
  headers: headers,
  data: JSON.stringify(ic)
}).done(function (result) {
  var id = result;
  //etc
});