使用jQuery Ajax将对象列表传递到MVC控制器方法中


113

我正在尝试使用jQuery的ajax()函数将对象数组传递到MVC控制器方法中。当我进入PassThing()C#控制器方法时,参数“ things”为空。我已经尝试使用List的类型作为参数,但这也不起作用。我究竟做错了什么?

<script type="text/javascript">
    $(document).ready(function () {
        var things = [
            { id: 1, color: 'yellow' },
            { id: 2, color: 'blue' },
            { id: 3, color: 'red' }
        ];

        $.ajax({
            contentType: 'application/json; charset=utf-8',
            dataType: 'json',
            type: 'POST',
            url: '/Xhr/ThingController/PassThing',
            data: JSON.stringify(things)
        });
    });
</script>

public class ThingController : Controller
{
    public void PassThing(Thing[] things)
    {
        // do stuff with things here...
    }

    public class Thing
    {
        public int id { get; set; }
        public string color { get; set; }
    }
}


3
您的数据是一个字符串,但是您的方法接受一个数组。更改您的方法以接受字符串,然后在方法中将其反序列化。
鲍勃·霍恩

2
您的代码是正确的。我对其进行了测试,并且使用MVC 4进行了测试。请提供更多数据以解决问题。
迭戈

这是好东西,但是如果您不仅需要传递字符串列表,还需要包括与字符串列表相关联的单独ID,该怎么办?就像群组ID,群组ID下的群组列表。
内森·麦卡斯克尔

Answers:


188

根据NickW的建议,我可以使用things = JSON.stringify({ 'things': things });下面的完整代码来使其正常工作。

$(document).ready(function () {
    var things = [
        { id: 1, color: 'yellow' },
        { id: 2, color: 'blue' },
        { id: 3, color: 'red' }
    ];      

    things = JSON.stringify({ 'things': things });

    $.ajax({
        contentType: 'application/json; charset=utf-8',
        dataType: 'json',
        type: 'POST',
        url: '/Home/PassThings',
        data: things,
        success: function () {          
            $('#result').html('"PassThings()" successfully called.');
        },
        failure: function (response) {          
            $('#result').html(response);
        }
    }); 
});


public void PassThings(List<Thing> things)
{
    var t = things;
}

public class Thing
{
    public int Id { get; set; }
    public string Color { get; set; }
}

我从中学到了两件事:

  1. 在ajax()函数中,绝对需要contentType和dataType设置。如果他们失踪了,那是行不通的。经过反复试验,我发现了这一点。

  2. 要将对象数组传递给MVC控制器方法,只需使用JSON.stringify({'things':Things})格式。

我希望这可以帮助其他人!


8
我遇到了同样的问题,并添加了contentType修复它。谢谢!
罗谢尔C

9
需要注意的两件事:JSON.stringify和指定'contentType'。
dinesh ygv 2014年

克鲁德。仍然不能为我工作。我的请求网址为http://localhost:52459/Sales/completeSale?itemsInCart=[{"ItemId":1,"Quantity":"1","Price":3.5}]Sales.completeSalepublic ActionResult completeSale(ItemInCart[] itemsInCart),注释为HttpGet
2015年

3
无论出于什么原因,我都不得不使用它data: JSON.stringify(things),
罗伯·斯科特

1
dataType没有必要。如果省略,ajax函数将根据返回数据进行计算

32

你不能只是这样做吗?

var things = [
    { id: 1, color: 'yellow' },
    { id: 2, color: 'blue' },
    { id: 3, color: 'red' }
];
$.post('@Url.Action("PassThings")', { things: things },
   function () {
        $('#result').html('"PassThings()" successfully called.');
   });

...并用

[HttpPost]
public void PassThings(IEnumerable<Thing> things)
{
    // do stuff with things here...
}

3
这应该是最好的答案。在这种情况下,不应使用

这对我不起作用。我正在使用[HttpPost] public int SaveResults(List <ShortDetail> model){}和$ .post(“ @ Url.Action(” SaveResults“,” Maps“)”,{model: dataItems},函数(结果){});
萨姆拉(Samra)

2
它为我工作。绝对是最好的答案。我不知道为什么Halcyon实施不起作用。已调用PassThings函数,但是即使在调用之前在javascript中填充了“ things”输入变量,该变量也是空的。
莱昂纳多·达加


12

我正在使用.Net Core 2.1 Web应用程序,在这里无法获得一个答案。我要么得到一个空白参数(如果根本调用了该方法),要么得到500个服务器错误。我开始尝试各种可能的答案,最后得到了可行的结果。

就我而言,解决方案如下:

脚本-字符串化原始数组(不使用命名属性)

    $.ajax({
        type: 'POST',
        contentType: 'application/json; charset=utf-8',
        url: mycontrolleraction,
        data: JSON.stringify(things)
    });

在控制器方法中,使用[FromBody]

    [HttpPost]
    public IActionResult NewBranch([FromBody]IEnumerable<Thing> things)
    {
        return Ok();
    }

失败包括:

  • 命名内容

    数据:{内容:节点},//服务器错误500

  • 没有contentType =服务器错误500

笔记

  • dataType尽管有一些答案说,也不需要,因为它用于响应解码(因此与此处的请求示例无关)。
  • List<Thing> 也适用于控制器方法

10

对于这一切,我都有一个完美的答案:我尝试了很多无法最终解决的解决方案,请在下面找到详细的答案:

       $.ajax({
            traditional: true,
            url: "/Conroller/MethodTest",
            type: "POST",
            contentType: "application/json; charset=utf-8",
            data:JSON.stringify( 
               [
                { id: 1, color: 'yellow' },
                { id: 2, color: 'blue' },
                { id: 3, color: 'red' }
                ]),
            success: function (data) {
                $scope.DisplayError(data.requestStatus);
            }
        });

控制者

public class Thing
{
    public int id { get; set; }
    public string color { get; set; }
}

public JsonResult MethodTest(IEnumerable<Thing> datav)
    {
   //now  datav is having all your values
  }

您应该有更多的赞誉:传统:在Jquery网站上推荐使用true的方式
DFTR

7

我可以使它起作用的唯一方法是将JSON作为字符串传递,然后使用进行反序列化。 JavaScriptSerializer.Deserialize<T>(string input),如果这是MVC 4的默认反序列化器,这是很奇怪的。

我的模型具有嵌套的对象列表,使用JSON数据可以得到的最好的结果是最上面的列表,其中包含正确数量的项目,但是项目中的所有字段均为null。

这种事情不应该那么难。

    $.ajax({
        type: 'POST',
        url: '/Agri/Map/SaveSelfValuation',
        data: { json: JSON.stringify(model) },
        dataType: 'text',
        success: function (data) {

    [HttpPost]
    public JsonResult DoSomething(string json)
    {
        var model = new JavaScriptSerializer().Deserialize<Valuation>(json);

要使此工作有效,请严格遵循Ajax调用的格式。
Graham Laight

4

这是您查询的工作代码,您可以使用它。

控制者

    [HttpPost]
    public ActionResult save(List<ListName> listObject)
    {
    //operation return
    Json(new { istObject }, JsonRequestBehavior.AllowGet); }
    }

javascript

  $("#btnSubmit").click(function () {
    var myColumnDefs = [];
    $('input[type=checkbox]').each(function () {
        if (this.checked) {
            myColumnDefs.push({ 'Status': true, 'ID': $(this).data('id') })
        } else {
            myColumnDefs.push({ 'Status': false, 'ID': $(this).data('id') })
        }
    });
   var data1 = { 'listObject': myColumnDefs};
   var data = JSON.stringify(data1)
   $.ajax({
   type: 'post',
   url: '/Controller/action',
   data:data ,
   contentType: 'application/json; charset=utf-8',
   success: function (response) {
    //do your actions
   },
   error: function (response) {
    alert("error occured");
   }
   });

2

用另一个对象包装您的对象列表,该对象包含的属性与MVC控制器期望的参数名称相匹配。重要的一点是对象列表周围的包装器。

$(document).ready(function () {
    var employeeList = [
        { id: 1, name: 'Bob' },
        { id: 2, name: 'John' },
        { id: 3, name: 'Tom' }
    ];      

    var Employees = {
      EmployeeList: employeeList
    }

    $.ajax({
        dataType: 'json',
        type: 'POST',
        url: '/Employees/Process',
        data: Employees,
        success: function () {          
            $('#InfoPanel').html('It worked!');
        },
        failure: function (response) {          
            $('#InfoPanel').html(response);
        }
    }); 
});


public void Process(List<Employee> EmployeeList)
{
    var emps = EmployeeList;
}

public class Employee
{
    public int Id { get; set; }
    public string Name { get; set; }
}

1
     var List = @Html.Raw(Json.Encode(Model));
$.ajax({
    type: 'post',
    url: '/Controller/action',
    data:JSON.stringify({ 'item': List}),
    contentType: 'application/json; charset=utf-8',
    success: function (response) {
        //do your actions
    },
    error: function (response) {
        alert("error occured");
    }
});

尝试使用此代码以使用ajax传递模型对象列表。模型代表IList <Model>。在控制器中使用IList <Model>来获取值。
Athul Nalupurakkal

0

如果您使用的是ASP.NET Web API,则应该通过 data: JSON.stringify(things)

并且您的控制器应如下所示:

public class PassThingsController : ApiController
{
    public HttpResponseMessage Post(List<Thing> things)
    {
        // code
    }
}

0

来自@veeresh的修改我

 var data=[

                        { id: 1, color: 'yellow' },
                        { id: 2, color: 'blue' },
                        { id: 3, color: 'red' }
                        ]; //parameter
        var para={};
        para.datav=data;   //datav from View


        $.ajax({
                    traditional: true,
                    url: "/Conroller/MethodTest",
                    type: "POST",
                    contentType: "application/json; charset=utf-8",
                    data:para,
                    success: function (data) {
                        $scope.DisplayError(data.requestStatus);
                    }
                });

In MVC



public class Thing
    {
        public int id { get; set; }
        public string color { get; set; }
    }

    public JsonResult MethodTest(IEnumerable<Thing> datav)
        {
       //now  datav is having all your values
      }

0

尝试从DataTable中的几个选定行发送一些数据到MVC操作时我做了什么:

HTML在页面的开头:

@Html.AntiForgeryToken()

(仅显示一行,从模型绑定):

 @foreach (var item in Model.ListOrderLines)
                {
                    <tr data-orderid="@item.OrderId" data-orderlineid="@item.OrderLineId" data-iscustom="@item.IsCustom">
                        <td>@item.OrderId</td>
                        <td>@item.OrderDate</td>
                        <td>@item.RequestedDeliveryDate</td>
                        <td>@item.ProductName</td>
                        <td>@item.Ident</td>
                        <td>@item.CompanyName</td>
                        <td>@item.DepartmentName</td>
                        <td>@item.ProdAlias</td>
                        <td>@item.ProducerName</td>
                        <td>@item.ProductionInfo</td>
                    </tr>
                }

启动JavaScript功能的按钮:

 <button class="btn waves-effect waves-light btn-success" onclick="ProcessMultipleRows();">Start</button>

JavaScript函数:

  function ProcessMultipleRows() {
            if ($(".dataTables_scrollBody>tr.selected").length > 0) {
                var list = [];
                $(".dataTables_scrollBody>tr.selected").each(function (e) {
                    var element = $(this);
                    var orderid = element.data("orderid");
                    var iscustom = element.data("iscustom");
                    var orderlineid = element.data("orderlineid");
                    var folderPath = "";
                    var fileName = "";

                    list.push({ orderId: orderid, isCustomOrderLine: iscustom, orderLineId: orderlineid, folderPath: folderPath, fileName : fileName});
                });

                $.ajax({
                    url: '@Url.Action("StartWorkflow","OrderLines")',
                    type: "post", //<------------- this is important
                    data: { model: list }, //<------------- this is important
                    beforeSend: function (xhr) {//<--- This is important
                      xhr.setRequestHeader("RequestVerificationToken",
                      $('input:hidden[name="__RequestVerificationToken"]').val());
                      showPreloader();
                    },
                    success: function (data) {

                    },
                    error: function (XMLHttpRequest, textStatus, errorThrown) {

                    },
                     complete: function () {
                         hidePreloader();
                    }
                });
            }
        }

MVC动作:

[HttpPost]
[ValidateAntiForgeryToken] //<--- This is important
public async Task<IActionResult> StartWorkflow(IEnumerable<WorkflowModel> model)

和C#中的MODEL:

public class WorkflowModel
 {
        public int OrderId { get; set; }
        public int OrderLineId { get; set; }
        public bool IsCustomOrderLine { get; set; }
        public string FolderPath { get; set; }
        public string FileName { get; set; }
 }

结论:

错误的原因:

"Failed to load resource: the server responded with a status of 400 (Bad Request)"

是属性: [ValidateAntiForgeryToken]用于MVC操作StartWorkflow

Ajax调用中的解决方案:

  beforeSend: function (xhr) {//<--- This is important
                      xhr.setRequestHeader("RequestVerificationToken",
                      $('input:hidden[name="__RequestVerificationToken"]').val());
                    },

要发送对象列表,您需要形成数据,例如示例(填充列表对象),以及:

数据:{模型:列表},

类型:“ post”,


0

这对我来说是正常的:

var things = [
    { id: 1, color: 'yellow' },
    { id: 2, color: 'blue' },
    { id: 3, color: 'red' }
];

$.ajax({
    ContentType: 'application/json; charset=utf-8',
    dataType: 'json',
    type: 'POST',
    url: '/Controller/action',
    data: { "things": things },
    success: function () {
        $('#result').html('"PassThings()" successfully called.');
    },
    error: function (response) {
        $('#result').html(response);
    }
});

大写字母“ C”中的“ ContentType”。

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.