再谈Jquery Ajax方法传递到action(补充)


Posted in Javascript onMay 12, 2014

之前写过一篇文章Jquery Ajax方法传值到action,本文是对该文的补充
假设 controller中的方法是如下:

public ActionResult ReadPerson(PersonModel model)  
        {  
            string s = model.ToString();  
            return Content(s);  
        } 
public ActionResult ReadPersons(List<PersonModel> model)  
        {  
            string result = "";  
            if (model == null) return Content(result);  
            foreach (var s in model)  
            {  
                result += s.ToString();  
                result += "-------------";  
            } 
            return Content(result);  
        }

其中PersonModel定义如下:

public class PersonModel  
    {  
        public int id  
        {  
            set;  
            get;  
        }  
        public string name  
        {  
            set;  
            get;  
        } 
        public int age  
        {  
            set;  
            get;  
        } 
        public bool gender  
        {  
            set;  
            get;  
        }  
        public string city  
        {  
            set;  
            get;  
        } 
        public override string ToString()  
        {  
            string s = string.Format(@"id:{0}  
name:{1}  
age:{2}  
gender:{3}  
city:{4}  
", id, name, age, gender, city);  
            return s;  
        }  
    }

那么controller方法分别接受单个model和一个model的List。采用通过ajax传递参数。
对于传递单个参数的情况,假设js代码如下:

var person = {  
               id: "001",  
               name: "zhangsan",  
               age: "20",  
               gender: true,  
               city: "shanghai"  
           }; 
var option = {  
               url: '/test/ReadPerson',  
               type: 'POST',  
               data: person,  
               dataType: 'html',  
               success: function (result) { alert(result); }  
           };  
$.ajax(option);

从chrome中截图可以看到如下:
再谈Jquery Ajax方法传递到action(补充)

传递的数据是一串Form数据,根据命名匹配的原则,也是可以取得数据的。
再谈Jquery Ajax方法传递到action(补充)

将option 的代码改成如下

var option = {  
               url: '/test/ReadPerson',  
               type: 'POST',  
               data: JSON.stringify(person),  
               dataType: 'html',  
               success: function (result) { alert(result); }  
           };  
$.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

看请求头的截图:

再谈Jquery Ajax方法传递到action(补充)

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

将将option 的代码改成如下

var option = {  
                url: '/test/ReadPerson',  
                type: 'POST',  
                data: person,  
                dataType: 'html',  
                 contentType: 'application/json',  
                 success: function (result) { alert(result); }  
                }; 

把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。

再谈Jquery Ajax方法传递到action(补充)

如果把processData设置为false。

再谈Jquery Ajax方法传递到action(补充)

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

因此,把option改成:

var option = {  
    url: '/test/ReadPerson',  
    type: 'POST',  
    data:JSON.stringify(person),  
    dataType: 'html',  
    contentType: 'application/json',  
    success: function (result) { alert(result); }  
}; 

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

再谈Jquery Ajax方法传递到action(补充)

对于较为简单是数据类型,有时候不指定contentType也能通过命名匹配传值。但是对于稍微复杂点的数据类型,有时指定contentType: 'application/json',处理起来更加方便。
如果一个controller里的action方法是接受一个List类型的参数,比如:
public ActionResult ReadPersons(List<PersonModel> model)
那么js中先构造这样的一个json对象的数组。如下

var persons = [{ 
                id: "001", 
                name: "zhangsan", 
                age: "20", 
                gender: true, 
                city: "shanghai" 
            }, 
            { 
                id: "002", 
                name: "lisi", 
                age: "21", 
                gender: false, 
                city: "beijing" 
            } 
            ];

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

var jsonp = { model: persons }; 
           var option = { 
               url: '/test/ReadPersons', 
               type: 'POST', 
               data: jsonp, 
               dataType: 'html', 
               success: function (result) { alert(result); } 
           };

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

再谈Jquery Ajax方法传递到action(补充)

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

再谈Jquery Ajax方法传递到action(补充)

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

var option = { 
              url: '/test/ReadPersons', 
              type: 'POST', 
              data: JSON.stringify(jsonp), 
              dataType: 'html', 
              success: function (result) { alert(result); } 
          };

再谈Jquery Ajax方法传递到action(补充)

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

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

var option = { 
    url: '/test/ReadPersons', 
    type: 'POST', 
    data: jsonp, 
    dataType: 'html', 
    contentType: 'application/json', 
    success: function (result) { alert(result); } 
};

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

再谈Jquery Ajax方法传递到action(补充)

再谈Jquery Ajax方法传递到action(补充)

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

var option = { 
                url: '/test/ReadPersons', 
                type: 'POST', 
                data: JSON.stringify(persons), 
                dataType: 'html', 
                contentType: 'application/json', 
                success: function (result) { alert(result); } 
            };

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

再谈Jquery Ajax方法传递到action(补充)

最后,此处再演示一个更复杂的参数类型,以便加深理解。
首先看一下Controller中的方法签名,TestClassB 和一个TestClassA的List。稍显复杂。

public ActionResult Fortest(TestClassB TB,List<TestClassA> TA) 
        { 
            string result = ""; 
            return Content(result); 
        }

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

public class TestClassA 
    { 
       public string a1 { set; get; } 
       public List<string> a2 { set; get; } 
    } 
    public class TestClassB 
    { 
        public string b1 { set; get; } 
        public InnerTestClassC ITCC { set; get; } 
        public class InnerTestClassC 
        { 
            public List<int> c1 { set; get; } 
        } 
    }

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

$("#btn").click(function () { 
            var jsondata = { TB: {}, TA: [] }; 
            jsondata.TB.b1 = "b1"; 
            jsondata.TB.ITCC = {}; 
            jsondata.TB.ITCC.c1 = new Array(1, 2, 3, 4); 
            var ta1 = {}; 
            ta1.a1 = "a1"; 
            ta1.a2 = new Array("a", "b", "x", "y"); 
           var ta2 = {}; 
            ta2.a1 = "a2"; 
            ta2.a2 = new Array("a2", "b2", "x2"); 
            jsondata.TA.push(ta1); 
            jsondata.TA.push(ta2); 
            var option = { 
                url: '/test/Fortest', 
                type: 'POST', 
                data: JSON.stringify(jsondata), 
                dataType: 'html', 
                contentType: 'application/json', 
                success: function (result) { alert(result); } 
            }; 
            $.ajax(option); 
        });

最终,发送出去的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串后,就能自动的匹配参数了。具体得到的参数如下截图:

再谈Jquery Ajax方法传递到action(补充)

再谈Jquery Ajax方法传递到action(补充)

总结:

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字符串作为发送数据,这样可读性更好,并且对于复杂的函数签名,也能起到很好的匹配。

本文出自 “一只博客” 博客

Javascript 相关文章推荐
JavaScript语言中的Literal Syntax特性分析
Mar 08 Javascript
JavaScript如何实现对数字保留两位小数一位自动补零
Dec 18 Javascript
浅谈jquery中next与siblings的区别
Oct 27 Javascript
JS控制TreeView的结点选择
Nov 11 Javascript
js 获取今天以及过去日期
Apr 11 Javascript
node.js中使用Export和Import的方法
Sep 18 Javascript
vue中echarts3.0自适应的方法
Feb 26 Javascript
axios实现简单文件上传功能
Sep 25 Javascript
JavaScript回调函数callback用法解析
Jan 14 Javascript
openlayers4实现点动态扩散
Aug 17 Javascript
vue-resource 拦截器interceptors使用详解
Jan 18 Vue.js
原生js拖拽功能制作滑动条实例代码
Feb 05 Javascript
Jquery Ajax方法传值到action的方法
May 11 #Javascript
json的定义、标准格式及json字符串检验
May 11 #Javascript
Jquery操作js数组及对象示例代码
May 11 #Javascript
Jquery+asp.net后台数据传到前台js进行解析的方法
May 11 #Javascript
如何防止回车(enter)键提交表单
May 11 #Javascript
js取整数、取余数的方法
May 11 #Javascript
浏览器窗口大小变化时使用resize事件对框架不起作用的解决方法
May 11 #Javascript
You might like
php使用ZipArchive提示Fatal error: Class ZipArchive not found in的解决方法
2014/11/04 PHP
php采用file_get_contents代替使用curl实例
2014/11/07 PHP
浅析php单例模式
2014/11/25 PHP
php实现支持中文的文件下载功能示例
2017/08/30 PHP
js 判断 enter 事件
2009/02/12 Javascript
JQuery EasyUI 对话框的使用方法
2010/10/24 Javascript
Js参数值中含有单引号或双引号问题的解决方法
2013/11/06 Javascript
jquery中ajax函数执行顺序问题之如何设置同步
2014/02/28 Javascript
表单提交前触发函数返回true表单才会提交
2014/03/11 Javascript
js检测浏览器版本、核心、是否移动端示例
2014/04/24 Javascript
使用JavaScript刷新网页的方法
2015/06/04 Javascript
JavaScript操作XML文件之XML读取方法
2015/06/09 Javascript
值得分享的Bootstrap Ace模板实现菜单和Tab页效果
2015/12/30 Javascript
jQuery解析返回的xml和json方法详解
2017/01/05 Javascript
jQuery基于ajax方式实现用户名存在性检查功能示例
2017/02/10 Javascript
jQuery实现多张图片上传预览(不经过后端处理)
2017/04/29 jQuery
重学JS 系列:聊聊继承(推荐)
2019/04/11 Javascript
移动端 Vue+Vant 的Uploader 实现上传、压缩、旋转图片功能
2019/06/10 Javascript
Python实现发送email的几种常用方法
2014/08/18 Python
使用Nginx+uWsgi实现Python的Django框架站点动静分离
2016/03/21 Python
python中os模块详解
2016/10/14 Python
带你了解python装饰器
2017/06/15 Python
Python编程实现两个文件夹里文件的对比功能示例【包含内容的对比】
2017/06/20 Python
使用Python代码实现Linux中的ls遍历目录命令的实例代码
2019/09/07 Python
Python使用pdb调试代码的技巧
2020/05/03 Python
详解python命令提示符窗口下如何运行python脚本
2020/09/11 Python
python 爬虫如何正确的使用cookie
2020/10/27 Python
PyChon中关于Jekins的详细安装(推荐)
2020/12/28 Python
CSS3轻松实现清新 Loading 效果的简单实例
2016/06/06 HTML / CSS
使用html2canvas将页面转成图并使用用canvas2image下载
2019/04/04 HTML / CSS
巴西备受欢迎的服装和生活方式品牌:FARM Rio
2020/02/04 全球购物
Genny意大利官网:意大利高级时装品牌
2020/04/15 全球购物
什么叫应用程序域?什么是受管制的代码?什么是强类型系统?什么是装箱和拆箱?
2016/08/13 面试题
营业用房租赁协议书
2014/11/26 职场文书
如何写辞职信
2015/05/13 职场文书
2015元旦感言
2015/12/09 职场文书