热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

再谈JqueryAjax方法传递到action【转载】

原创作品,允许转载,转载时请务必以超链接形式标明文章原始出处、作者信息和本声明。否则将追究法律责任。http:cnn237111.blog.51cto.com23591449844

原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 、作者信息和本声明。否则将追究法律责任。http://cnn237111.blog.51cto.com/2359144/984466

之前写过一篇文章Jquery Ajax方法传值到action,本文是对该文的补充。

假设 controller中的方法是如下:

  1. public ActionResult ReadPerson(PersonModel model) 
  2.         { 
  3.             string s = model.ToString(); 
  4.             return Content(s); 
  5.         }
  6.  
  7. public ActionResult ReadPersons(List model) 
  8.         { 
  9.             string result = ""; 
  10.             if (model == null) return Content(result); 
  11.             foreach (var s in model) 
  12.             { 
  13.                 result += s.ToString(); 
  14.                 result += "-------------"; 
  15.             }
  16.             return Content(result); 
  17.         }

其中PersonModel定义如下:

  1. public class PersonModel 
  2.     { 
  3.         public int id 
  4.         { 
  5.             set; 
  6.             get; 
  7.         } 
  8.         public string name 
  9.         { 
  10.             set; 
  11.             get; 
  12.         }
  13.  
  14.         public int age 
  15.         { 
  16.             set; 
  17.             get; 
  18.         }
  19.  
  20.         public bool gender 
  21.         { 
  22.             set; 
  23.             get; 
  24.         } 
  25.         public string city 
  26.         { 
  27.             set; 
  28.             get; 
  29.         }
  30.  
  31.         public override string ToString() 
  32.         { 
  33.             string s = string.Format(@"id:{0} 
  34. name:{1} 
  35. age:{2} 
  36. gender:{3} 
  37. city:{4} 
  38. ", id, name, age, gender, city); 
  39.             return s; 
  40.         } 
  41.     }

那么controller方法分别接受单个model和一个model的List。采用通过ajax传递参数。

对于传递单个参数的情况,假设js代码如下:

  1. var person = { 
  2.                id: "001", 
  3.                name: "zhangsan", 
  4.                age: "20", 
  5.                gender: true, 
  6.                city: "shanghai" 
  7.            };
  8.  
  9. var option = { 
  10.                url: ‘/test/ReadPerson‘, 
  11.                type: ‘POST‘, 
  12. data: person, 
  13.                dataType: ‘html‘, 
  14.                success: function (result) { alert(result); } 
  15.            }; 
  16. $.ajax(option);

从chrome中截图可以看到如下:,

传递的数据是一串Form数据,根据命名匹配的原则,也是可以取得数据的。,


将option 的代码改成如下

  1. var option = { 
  2.                url: ‘/test/ReadPerson‘, 
  3.                type: ‘POST‘, 
  4.                data: JSON.stringify(person), 
  5.                dataType: ‘html‘, 
  6.                success: function (result) { alert(result); } 
  7.            }; 
  8. $.ajax(option);

其中JSON.stringify方法签名为 stringify ( value [ , replacer [ , space ] ] ),根据ECMA-262标准stringify 函数返回的是JSON格式的字符串。它可以有3个参数。摘抄如下:

The stringify function returns a String in JSON format representing an ECMAScript value. It can take three parameters. The first parameter is required. The value parameter is an ECMAScript value, which is usually an object or array, although it can also be a String, Boolean, Number or null. The optional replacer parameter is either a function that alters the way objects and arrays are stringified, or an array of Strings and Numbers that acts as a white list for selecting the object properties that will be stringified. The optional space parameter is a String or Number that allows the result to have white space injected into it to improve human readability.

默认的ContentType的属性值是"application/x-www-form-urlencoded"

引自http://www.w3.org/TR/html401/interact/forms.html#adef-enctype

看请求头的截图:

,

因此,传递到controller的是一个json字符串,MVC根据命名匹配也是可以获得到参数的值。


将将option 的代码改成如下

  1. var option = { 
  2.                 url: ‘/test/ReadPerson‘, 
  3.                 type: ‘POST‘, 
  4.                 data: person, 
  5.                 dataType: ‘html‘, 
  6.                  contentType: ‘application/json‘, 
  7.                  success: function (result) { alert(result); } 
  8.                 }; 

把contentType改成json格式,那么得到的是出错的信息。

虽然person是json对象,但是jquery中的ajax,data会自动的被转换成查询字符串格式key1=value1&key2=value2这种形式,很显然这种形式不是json格式,因此会出错。

要避免转换成查询字符串格式,只需要设置processData为fasle即可。processData默认是true。

这里需要注意的是:当指定了contentType的时候,数据将不再按照Form Data的形式提交了,而是变成Request Data的形式提交。可以从图上的Request Header中看出。需要注意的是,Form Data提交的数据可以由FormCollection获得到。Request Data方式提交的则不能通过FormCollection获得。

如果把processData设置为默认值true。

,

如果把processData设置为false。

,

以上两种方式,按照application/json的类型传给都会失败,因为json是基于文本的格式,上面两种方式传递的都不是json文本。因此会出错。


因此,把option改成:

  1. var option = { 
  2.     url: ‘/test/ReadPerson‘, 
  3.     type: ‘POST‘, 
  4.     data:JSON.stringify(person), 
  5.     dataType: ‘html‘, 
  6.     contentType: ‘application/json‘, 
  7.     success: function (result) { alert(result); } 
  8. }; 

则传递的就是json文本,因此根据命名匹配,就能获得值了。

,


对于较为简单是数据类型,有时候不指定contentType也能通过命名匹配传值。但是对于稍微复杂点的数据类型,有时指定contentType: ‘application/json‘,处理起来更加方便。

如果一个controller里的action方法是接受一个List类型的参数,比如:

public ActionResult ReadPersons(List model)

那么js中先构造这样的一个json对象的数组。如下

  1. var persOns= [{
  2.                 id: "001",
  3.                 name: "zhangsan",
  4.                 age: "20",
  5.                 gender: true,
  6.                 city: "shanghai"
  7.             },
  8.             {
  9.                 id: "002",
  10.                 name: "lisi",
  11.                 age: "21",
  12.                 gender: false,
  13.                 city: "beijing"
  14.             }
  15.             ];

单纯一个数组传递是作为data传递是,Form Data也是无法识别出的。因此把这个数组再次组成一个json形式。如下:其中json的键值用model是为了能和controller中的参数名相同,可以匹配。

  1. var jsOnp= { model: persons };
  2.            var option = {
  3.                url: ‘/test/ReadPersons‘,
  4.                type: ‘POST‘,
  5.                data: jsonp,
  6.                dataType: ‘html‘,
  7.                success: function (result) { alert(result); }
  8.            };

由于未指定contentType,因此是默认的application/x-www-form-urlencoded。此时是按照Form Data的方式传递的,

,

可以从截图中看到。但是这种格式的数据,controller中只能获得指定model用2个元素,无法获得元素中属性的值。

,

如果把data改成JSON.stringify(jsonp),如下:    

  1. var option = {
  2.               url: ‘/test/ReadPersons‘,
  3.               type: ‘POST‘,
  4.               data: JSON.stringify(jsonp),
  5.               dataType: ‘html‘,
  6.               success: function (result) { alert(result); }
  7.           }; 

,

那么传递过去的Form Data是一串字符串,controller跟无法识别出这个东西,因此获不到值。如果仅仅设置contentType: ‘application/json‘,而传递的又不是json格式的数据,如下:

  1. var option = {
  2.     url: ‘/test/ReadPersons‘,
  3.     type: ‘POST‘,
  4.     data: jsonp,
  5.     dataType: ‘html‘,
  6.     contentType: ‘application/json‘,
  7.     success: function (result) { alert(result); }
  8. };

因为jquery的ajax方法会把data转换成查询字符串,因此就变成如下的样子。这串文本当然不符合json格式,因此会出现下面的错误。

,

,

如果设置contentType: ‘application/json‘,并且设置data: JSON.stringify(persons),如下:

  1. var option = {
  2.                 url: ‘/test/ReadPersons‘,
  3.                 type: ‘POST‘,
  4.                 data: JSON.stringify(persons),
  5.                 dataType: ‘html‘,
  6.                 contentType: ‘application/json‘,
  7.                 success: function (result) { alert(result); }
  8.             };

那么可以获得到真正完整的json数据了

,


最后,此处再演示一个更复杂的参数类型,以便加深理解。

首先看一下Controller中的方法签名,TestClassB 和一个TestClassA的List。稍显复杂。

  1. public ActionResult Fortest(TestClassB TB,List TA)
  2.         {
  3.             string result = "";
  4.             return Content(result);
  5.         }

再看TestClassA和TestClassB,更显复杂。但是结构要清晰的话,也不是很难。

  1. public class TestClassA
  2.     {
  3.        public string a1 { set; get; }
  4.        public List a2 { set; get; }
  5.     }
  6.     public class TestClassB
  7.     {
  8.         public string b1 { set; get; }
  9.         public InnerTestClassC ITCC { set; get; }
  10.         public class InnerTestClassC
  11.         {
  12.             public List c1 { set; get; }
  13.         }
  14.     }

看js代码:逐步的构造出一个json格式。

  1. $("#btn").click(function () {
  2.             var jsOndata= { TB: {}, TA: [] };
  3.             jsondata.TB.b1 = "b1";
  4.             jsondata.TB.ITCC = {};
  5.             jsondata.TB.ITCC.c1 = new Array(1, 2, 3, 4);
  6.             var ta1 = {};
  7.             ta1.a1 = "a1";
  8.             ta1.a2 = new Array("a", "b", "x", "y");
  9.            var ta2 = {};
  10.             ta2.a1 = "a2";
  11.             ta2.a2 = new Array("a2", "b2", "x2");
  12.             jsondata.TA.push(ta1);
  13.             jsondata.TA.push(ta2);
  14.             var option = {
  15.                 url: ‘/test/Fortest‘,
  16.                 type: ‘POST‘,
  17.                 data: JSON.stringify(jsondata),
  18.                 dataType: ‘html‘,
  19.                 contentType: ‘application/json‘,
  20.                 success: function (result) { alert(result); }
  21.             };
  22.             $.ajax(option);
  23.         });

最终,发送出去的json字符串如下:

{"TB":{"b1":"b1","ITCC":{"c1":[1,2,3,4]}},"TA":[{"a1":"a1","a2":["a","b","x","y"]},{"a1":"a2","a2":["a2","b2","x2"]}]}

Controller接收到这个json串后,就能自动的匹配参数了。具体得到的参数如下截图:

,

,


总结:

1.不指定contentType的话,默认都是application/x-www-form-urlencoded方式发送。此时即便发送的是json格式的数据,默认情况下,jquery的ajax也会把他转为查询字符串的形式(可以通过修改ajax参数修改),以FormData的形式发送出去。

2.不指定contentType的时候,如果controller中的方法签名比较简单,那么即便是FormData形式的数据也能由MVC的命名匹配规则获取到数据。

3.指定contentType为‘application/json‘时候,发送的数据必须是符合json规范的字符串。通常,使用 JSON.stringify(jsondata)有较好的可读性,可以获得一个json字符串。当然,不是必须的。使用拼接的字符串,只要是符合json规范的,也是可以发送的。

4.如果contentType为‘application/json‘时,发送的data不是符合json规范的字符串,则会出错。

5.通常情况下,尽量指定contentType为‘application/json‘,并且发送json字符串作为发送数据,这样可读性更好,并且对于复杂的函数签名,也能起到很好的匹配。

本文出自 “一只博客” 博客,请务必保留此出处http://cnn237111.blog.51cto.com/2359144/984466

再谈Jquery Ajax方法传递到action 【转载】


推荐阅读
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 后台获取视图对应的字符串
    1.帮助类后台获取视图对应的字符串publicclassViewHelper{将View输出为字符串(注:不会执行对应的ac ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 基于layUI的图片上传前预览功能的2种实现方式
    本文介绍了基于layUI的图片上传前预览功能的两种实现方式:一种是使用blob+FileReader,另一种是使用layUI自带的参数。通过选择文件后点击文件名,在页面中间弹窗内预览图片。其中,layUI自带的参数实现了图片预览功能。该功能依赖于layUI的上传模块,并使用了blob和FileReader来读取本地文件并获取图像的base64编码。点击文件名时会执行See()函数。摘要长度为169字。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
author-avatar
彭伟波2013
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有