WebAPI POST [FromBody]没有绑定

时间:2013-05-21 08:52:08

标签: c# asp.net-web-api

我将JSON发布到WebAPI控制器,但模型上的属性未受约束。

public void Post([FromBody] Models.Users.User model) {
    throw new Exception(model.Id.ToString());
}

原始请求如下:

POST http://diva2.local/siteapi/User HTTP/: diva2.local
Connection: keep-alive
Content-Length:: application/json, text/plain, */*
Origin: http://diva2.local
X-Requested-With: XMLHttpRequest
User-Agent: Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/537.31 (KHTML, like Gecko) Chrome/26.0.: application/json;charset=UTF: http://diva2.local/Users
Accept-Encoding: gzip,deflate,sdch
Accept-Language: en-US,en;q=: ISO-8859-1,utf-8;q=0.7,*;q=: .ASPXAUTH=4; __RequestVerificationToken=Rp_hUysjwCjmsxw2

{"Id":3,"FirstName":"DIVA2","LastName":"User1","Username":"diva1","IsApproved":true,"IsOnlineNow":true,"IsChecked":true}

我能找到的每个例子都告诉我这应该有用,但是model.Id == null

但是,如果我将JSON更改为:

{User: {"Id":3,"FirstName":"DIVA2","LastName":"User1","Username":"diva1","IsApproved":true,"IsOnlineNow":true,"IsChecked":true}}

一切都正确绑定。

这似乎不正确。我想我可以接受JObject作为参数,并手动绑定它,但感觉上面应该只是工作(tm)?

更新

我已经更改了返回模型的方法,但仍然收到null。

public Models.Users.User Post(Models.Users.User user) {
    return user;
}

回复:

HTTP/1.1 200 OK
Cache-Control: no-cache
Pragma: no-cache
Content-Type: application/json; charset=utf-8
Expires: -1
Server: Microsoft-IIS/8.0
X-AspNet-Version: 4.0.30319
X-MiniProfiler-Ids: ["a0fab514-d725-4d8f-9021-4931dc06ec4a","fdeeb9a8-9e36-41d8-91d3-5348e880e193","c1b4cc86-d7c3-4497-8699-baac9fa79bf1"]
X-Powered-By: ASP.NET
Date: Tue, 21 May 2013 09:06:00 GMT
Content-Length: 4

null

5 个答案:

答案 0 :(得分:24)

您的请求中缺少内容类型标题。

不幸的是,即使你检查过ModelState,我们也不会抛出任何错误信息。但是,好消息是我们即将发布的版本已修复此行为,您会看到基于状态代码的415响应。

Web API要求Content-Type标头找出正确的格式化程序,以将主体反序列化为操作上的参数。

答案 1 :(得分:17)

以下对我来说非常好:

型号:

public class User
{
    public int Id { get; set; }
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public string Username { get; set; }
    public bool IsApproved { get; set; }
    public bool IsOnlineNow { get; set; }
    public bool IsChecked { get; set; }
}

控制器:

public class ValuesController : ApiController
{
    public User Post(User user)
    {
        return user;
    }
}

请求:

POST http://example.com/api/values HTTP/1.1
Connection: keep-alive
Accept: application/json, text/plain, */*
Content-Type: application/json;charset=UTF-8
Host: localhost:8816
Content-Length: 125

{"Id":3,"FirstName":"DIVA2","LastName":"User1","Username":"diva2user1","IsApproved":true,"IsOnlineNow":true,"IsChecked":true}

响应:

HTTP/1.1 200 OK
Cache-Control: no-cache
Pragma: no-cache
Content-Type: application/json; charset=utf-8
Expires: -1
Server: Microsoft-IIS/8.0
Date: Tue, 21 May 2013 08:59:02 GMT
Content-Length: 125

{"Id":3,"FirstName":"DIVA2","LastName":"User1","Username":"diva2user1","IsApproved":true,"IsOnlineNow":true,"IsChecked":true}

正如您所看到的,在请求JSON有效负载中没有User前缀的情况下,所有内容都很好并且花花公子。

请注意model.Id,因为id如果您在路线定义中将其用作路线的一部分,则可能具有特殊含义。不要混淆两件事(路线参数和来自请求体有效载荷的参数)。

答案 2 :(得分:11)

我的用户模型没有无参数构造函数。

答案 3 :(得分:2)

我发现我的班级在所有属性上都有内部设置器。

改变     public <property> {get; internal set;} 至     public <property> {get; set;}

解决了我的问题

答案 4 :(得分:0)

我在课堂上有SerializablexmlElement标签。即使我在请求中将JSON指定为内容类型并传递了JSON字符串,也要等到将其删除后才能使用...似乎也只有serializable属性引起了问题。