Ajax与数据传输

背景

在没有ajax之前,前端与后台传数据都是靠表单传输,使用表单的方法传输数据有一个比较大的问题就是每次提交数据都会刷新页面,用户体验很不友好,解决方案有的是采用iframe,表单放在iframe中,不用刷新母页面,有的是在js里面重定位,这些方法都比较繁琐。所以人们发明了ajax这种异步通信方式。

ajax的应用

首先需要说明一点的是ajax是异步通信的,客户端向服务器请求一条数据后就不管了,继续执行下面的代码。接着是服务器,服务器那边接收到请求后就开始处理客户端的请求,处理完毕后再将数据发给客户端。然后客户端这时才继续执行相应的函数。
利用异步通信这一点,我们可以实现不用刷新页面就实现页面更新。利用ajax,我们可以实现很多有趣的功能。比如百度搜索的提示。当用户在搜索框输入字符时,下方会出现提示语。


百度的ajax

按下f12,选择network,会发现其实每次我们输入一个字符,百度的服务器那边就会通过ajax发送一条数据过来,这些数据就是下方的提示语。

如何使用ajax

这是一个简单的验证登录的js代码

            var username=document.getElementById("name").value;
                        var password=document.getElementById("pwd").value;
                        var xhr=null;
                        xhr=new XMLHttpRequest();//新建一个XMLHttpRequest对象
                        xhr.open("get",'./check.php',true);//初始化请求
                        xhr.onreadystatechange=function () {//定义数据返回后的回调函数
                            if(xhr.readyState==4){
                                if(xhr.status===200){
                                    var data=xhr.responseText;
                                    if(data==1){
                                        result.innerText="登录成功";
                                    }
                                    else if(data===2){
                                        result.innerText="登录失败";
                                    }
                                }
                            }
                        };
                        xhr.send(null);//xhr.send(null);
                        
                    };

ajax通信的步骤如下:

  1. 新建一个XMLHttpRequest对象。
  2. open方法表示初始化请求,此时并没有发送。
  3. 定义数据返回后的回调函数,里面的代码在readystatechange值改变的时候执行。
  4. 发送请求。

下面对上面提到的方法,参数做一个介绍。

open方法

用途:初始化请求。
参数一:请求方式,get或者post。
参数二:请求路径,数据被传输的目的地地址。
参数三:true:异步请求。false:同步请求。

Get请求

  • 将表单数据以名称/值对的形式附加到 URL 中
  • URL 的长度是有限的(大约 3000 字符)
  • 绝不要使用 GET 来发送敏感数据!(在 URL 中是可见的)
  • 对于用户希望加入书签的表单提交很有用
  • GET 更适用于非安全数据,比如在 Google 中查询字符串
    通过get的方式请求数据,那么对于send()方法,参数为null就可以。

get请求中如果有中文,可能会报错。所以,我们一般会对参数重新编码。

var url="open.php?username="+encodeURI(username)+"&password="+password;

ajax使用get方式是如何传输数据?
get方式的参数是放在url里面传输的,所以我们只要把参数放到url后面,拼接url字符串就可以了。

var url = './check.php?username='+username+"&password="+password;

Post请求

  • 将表单数据附加到 HTTP 请求的 body 内(数据不显示在 URL 中)
  • 没有长度限制
  • 通过 POST 提交的表单不能加入书签

post请求传输数据的时候,数据是作为send方法的参数传输的。

var param = 'username='+username+'&password='+password;
xhr.send(param);

一般数据接收方,也就是后台对于post请求,会默认其数据类型是表单数据类型,所以需要我们设置一下post请求的数据格式,否则我们提交的数据会因为编码格式不对,导致后台取不到我们提交的数据。

xhr.setRequestHeader("Content-Type","application/x-www-form-urlencoded; charset=gb2312");

readyState

readyState只有5个值{0,1,2,3,4},只读不能写。
0:XMLHttpRequest对象创建完成。
1:XMLHttpRequest对象初始化完成。
2:请求已经发送。
3:服务器已经返回了数据(但是还没有被解析,可能只一段http报文)。
4:数据解析已经完成。

status

status状态码,来自于http协议。详细的状态码以及意义可以参考http协议。推荐一本讲http协议的书籍,《图解http》,讲的比较简单,通熟易懂。
比较常见的:
200:成功
404:页面不存在
503:页面存在,但有语法错误

ResponseText和ResponseXml

数据返回的格式。
ResponseText表示返回的是一个字符串,也就是一个JSON。对象,和localstroge有点像。不管文件内容是什么都直接返回字符串。
ResponseXml表示返回的是一个xml格式的文档,如果返回的数据中有不符合xml格式的,则不会返回。比如文档内容如下:

<book>
        <name>红楼梦</name>
        <category>文学</category>
        <desc>宝哥哥与林妹妹的爱情史</desc>
    </book> 
    helloworld

helloworld没有被标签闭合,不符合xml格式。
var data1=xhr.responseXML;打印data1会发现结果中并没有helloworld,因为它不符合xml格式,所以不会被传输过来。

url中的参数传递格式

url是一个地址,当我们需要传递参数的时候,首先需要在url后添加一个?号,然后后面跟上我们的参数。比如username="哈士奇",参数之间要用&号连接。比如username="哈士奇"&ppassword="我爱萨摩耶"。完整的例子如下:http://www.whutyzy.cc/index.php?username=哈士奇&ppassword=我爱萨摩耶

jquery中的ajax

每次发送请求,都要写上这么一段代码:

            var xhr=null;
            xhr=new XMLHttpRequest();
            xhr.open("get",url,true);
            xhr.send(null);
            xhr.onreadystatechange=function () {
                if(xhr.readyState===4){
                    if(xhr.status===200){
                        dosomething();
                    }
                }
            };

其中大部分的代码都一样,所以我们完全可以抽象成一个方法,所幸jquery方法以及帮我们封装好了一系列的ajax方法,我们只要直接调用就可以了。
使用jquery发送ajax请求的话,只要一行代码就可以了。

$.ajax({url:"test.js",dataType:"json"});

更多详细的信息请参考jquery-ajax教程。

jsonp

在讲jsonp是什么之前,我们先看下这么一个情况,我们想写一个天气预报的实时页面,数据肯定是从其他网站来的,比如url:"http://tq.360.cn/api/weatherquery/querys?app=tq360&code=$code&_jsonp=renderData&_=1429595284544"这个接口就提供了我们需要的天气数据。所以我们只需要使用ajax发送请求到这个网站上,再接收天气数据在页面上渲染就可以了。但是当我们写好代码运行的时候会报错,提示No 'Access-Control-Allow-Origin' header is present on the requested resource. Origin 'http://localhost:63342' is therefore not allowed access.
大概意思是无法接收请求,原因是发生了跨域,也就是我们访问了其他域名,出于安全原因,我们不会接收其他域名发送过来的文件数据。对于这个跨域问题,有不同的解决方案,我们使用的方案是jsonp。
jsonp到底是什么?不要着急,继续往下看。
虽然浏览器不让我们接收其他网站的数据,但是允许我们加载其他网站上的script文件。比如比较常见的引入jquery:

<script src="https://cdn.bootcss.com/jquery/3.2.1/jquery.js"></script>

所以我们完全可以通过script的方式来加载数据。

<body>
<script src="https://cdn.bootcss.com/我爱哈士奇.php"></script>
</body>

假设这个我爱哈士奇.php的内容如下:

echo "helloworld";//输出helloworld

那么页面中<script src="https://cdn.bootcss.com/我爱哈士奇.php"></script>加载完后会返回helloworld。在页面中就是

<body>
<script>
helloworld
</script>
</body>

通过这样的方式,我们成功的获取了其他网站上的数据。
现在回到jsonp是什么这个问题上,jsonp就是值通过加载script文件这种方式解决跨域问题的一种方案(有问题,不过暂时这么理解)。但是实际上jsonp比上面这个例子稍微复杂一点。首先我们看一下html文件:

<!doctype html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>jsonp</title>
</head>
<body>
    <script type="text/javascript">
    function cb(data){
        console.log(data);
    }
    </script>
    <script type="text/javascript" src="https://cdn.bootcss.com/我爱哈士奇.php?_jsonp=cb"></script>
</body>
</html>

在主体下面我们引入了一个我爱哈士奇的php文件,这个php文件内容如下:

<?php 
$callback = $_GET['_jsonp'];
$arr = array("大金毛","哈士奇","短腿柯基");
echo $callback."(".json_encode($arr).")";

 ?>

这段php代码用js写的话代码如下

var callback=GET['_jsonp'];
var arr="哈士奇";
console.log(callback+"("+arr+")");

大概意思是获取url传过来的参数_jsonp的值并赋值给callback,然后再拼接字符串,拼成callback(arr)。然后再将拼好的字符串加载到页面上。
最后页面中js代码段内容如下

function cb(data){
        console.log(data);
}
cb("哈士奇");

这一段绕来绕去做了个什么?其实就是将传回来的数据放到之前定义好的函数中执行。因为后台的获取这个函数名是根据url传过来的参数取出来的,所以这个参数名_jsonp最好统一,方便后台程序员编写接口。
到这里,我们再看下jsonp的定义与背景:
1、一个众所周知的问题,Ajax直接请求普通文件存在跨域无权限访问的问题,甭管你是静态页面、动态网页、web服务、WCF,只要是跨域请求,一律不准;
  2、不过我们又发现,Web页面上调用js文件时则不受是否跨域的影响(不仅如此,我们还发现凡是拥有”src”这个属性的标签都拥有跨域的能力,比如<script>、<img>、<iframe>
  3、于是可以判断,当前阶段如果想通过纯web端(ActiveX控件、服务端代理、属于未来的HTML5之Websocket等方式不算)跨域访问数据就只有一种可能,那就是在远程服务器上设法把数据装进js格式的文件里,供客户端调用和进一步处理;
  4、恰巧我们已经知道有一种叫做JSON的纯字符数据格式可以简洁的描述复杂数据,更妙的是JSON还被js原生支持,所以在客户端几乎可以随心所欲的处理这种格式的数据;
  5、这样子解决方案就呼之欲出了,web客户端通过与调用脚本一模一样的方式,来调用跨域服务器上动态生成的js格式文件(一般以JSON为后缀),显而易见,服务器之所以要动态生成JSON文件,目的就在于把客户端需要的数据装入进去。
  6、客户端在对JSON文件调用成功之后,也就获得了自己所需的数据,剩下的就是按照自己需求进行处理和展现了,这种获取远程数据的方式看起来非常像AJAX,但其实并不一样。
  7、为了便于客户端使用数据,逐渐形成了一种非正式传输协议,人们把它称作JSONP,该协议的一个要点就是允许用户传递一个callback参数给服务端,然后服务端返回数据时会将这个callback参数作为函数名来包裹住JSON数据,这样客户端就可以随意定制自己的函数来自动处理返回数据了。

当然,jquery中也提供了封装好的jsonp方法。最后再提一句,一般不会选择在页面上直接引入<script>这种方式,而是在js中动态生成<script>节点。

参考文献:

说说JSON和JSONP,也许你会豁然开朗

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容

  • AJAX 原生js操作ajax 1.创建XMLHttpRequest对象 var xhr = new XMLHtt...
    碧玉含香阅读 3,195评论 0 7
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,650评论 18 139
  • 第一章 入门 基本功能:访问和操作 dom 元素,控制页面样式,对页面的事件处理,与ajax完美结合,有丰富的插件...
    X_Arts阅读 1,045评论 0 2
  • Ajax 模块也是经常会用到的模块,Ajax 模块中包含了 jsonp 的现实,和 XMLHttpRequest ...
    对角另一面阅读 600评论 0 1
  • 你我如同 春去秋来 纵有相同的轨迹却不得相见
    小五姑凉阅读 182评论 0 1