ajax现在是一种非常流行的技术,现在虽然可以利用JQuery或者一些第三方插件甚至微软提供的一些控件可以方面的实现ajax功能,但是明白其原理也是非常重要的,下面是来使用纯javascript实现获取服务器端的功能来展示如何使用纯javascript实现ajax功能,以弄清其原理.

在前端页面开发的过程中,经常使用到Ajax请求,异步提交表单数据,或者异步刷新页面。

一般来说,使用Jquery中的$.ajax,$.post,$.getJSON,非常方便,但是有的时候,我们只需要ajax功能,这样引入Jquery比较不划算。

所以接下来便用原生JavaScrpit实现一个简单的Ajax请求,并说明ajax请求中的跨域访问问题,以及多个ajax请求的数据同步问题。

JavaScript实现Ajax异步请求

简单的ajax请求实现

Ajax请求的原理是创建一个XMLHttpRequest对象,使用这个对象来进行异步发送请求,具体实现参考下面代码:function ajax(option) {

// 创建一个 XMLHttpRequest 对象

var xhr = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject("Microsoft.XMLHTTP"),

requestData = option.data,

requestUrl = option.url,

requestMethod = option.method;

// 如果是GET请求,需要将option中的参数拼接到URL后面

if ('POST' != requestMethod && requestData) {

var query_string = '';

// 遍历option.data对象,构建GET查询参数

for(var item in requestData) {

query_string += item + '=' + requestData[item] + '&';

}

// 注意这儿拼接的时候,需要判断是否已经有 ?

requestUrl.indexOf('?') > -1

? requestUrl = requestUrl + '&' + query_string

: requestUrl = requestUrl + '?' + query_string;

// GET 请求参数放在URL中,将requestData置为空

requestData = null;

}

// ajax 请求成功之后的回调函数

xhr.onreadystatechange = function () {

// readyState=4表示接受响应完毕

if (xhr.readyState == ("number" == typeof XMLHttpRequest.DONE ? XMLHttpRequest.DONE : 4)) {

if (200 == xhr.status) { // 判断状态码

var response = xhr.response || xhr.responseText || {}; // 获取返回值

// if define success callback, call it, if response is string, convert it to json objcet

console.log(response);

option.success && option.success(response); // 调用回调函数处理返回数据

// 可以判断返回数据类型,对数据进行JSON解析或者XML解析

// option.success && option.success('string' == typeof response ? JSON.parse(response) : response);

} else {

// if define error callback, call it

option.error && option.error(xhr, xhr.statusText);

}

}

};

// 发送ajax请求

xhr.open(requestMethod, requestUrl, true);

// 请求超时的回调

xhr.ontimeout = function () {

option.timeout && option.timeout(xhr, xhr.statusText);

};

// 定义超时时间

xhr.timeout = option.timeout || 0;

// 设置响应头部,这儿默认设置为json格式,可以定义为其他格式,修改头部即可

xhr.setRequestHeader && xhr.setRequestHeader('Content-Type', 'application/json;charset=utf-8');

xhr.withCredentials = (option.xhrFields || {}).withCredentials;

// 这儿主要用于发送POST请求的数据

xhr.send(requestData);

}

上面的代码中有详细的注释,ajax的原理很简单,总的来说就是使用XMLHttpRequest对象来发送数据。这儿对这个对象进行补充说明。

XMLHttpRequest对象的基本属性

readyState属性有五个状态值:

0:是uninitialized:未初始化。已经创建了XMLHttpRequest对象但是未初始化。

1:是loading:已经开始准备好要发送了。

2:是loaded,:已经发送,但是还没有收到响应。

3:是interactive:正在接受响应,但是还没接收完。

4:是completed:接受响应完毕。

responseText:服务器返回的响应文本。只有当readyState>=3的时候才有值。当readyState=3,返回的响应文本不完整,只有readyState=4,接收到完整的响应文本。

responseXML:响应信息是xml,可以解析为Dom对象。

status:服务器的Http状态码,若是200,则表示OK,404,表示为未找到。

statusText:服务器http状态码的文本。比如OK,Not Found。

XMLHttpRequest对象的基本方法

open(method,url,asyn):打开XMLHttpRequest对象。其中method方法有get,post,delete,put。url是请求资源的地址。第三个参数表示是否使用异步。默认情况是true,因为Ajax的特点就是异步传送。若使用同步则false。

send(body):发送请求Ajax。其中发送的内容可以是需要的参数,若是没有参数,直接send(null)

使用方法

直接调用上面定义的ajax函数,传送相应的选项和参数即可。ajax({

url: '/post.php',

data: {

name: 'uusama',

desc: 'smart'

},

method: 'GET',

success: function(ret) {

console.log(ret);

}

});

跨域请求问题

使用ajax请求的时候,一定要注意一个问题:跨域请求。

在没有使用特殊手段的情况下,跨域请求:请求其他域名和端口下的URL资源的时候,会报 Access-Control-Allow-Origin 相关的错误。其主要原因是浏览器的同源策略限制,浏览器规定不能跨域请求资源。

解决办法

下面简单的提一下一些解决方案。

在ajax头部添加允许跨域请求的header,这种方式还需要服务端配合添加允许跨域请求的头部才可以。下面是PHP添加允许POST请求跨域头部的PHP示例:// 指定允许其他域名访问

header('Access-Control-Allow-Origin:*');

// 响应类型

header('Access-Control-Allow-Methods:POST');

// 响应头设置

header('Access-Control-Allow-Headers:x-requested-with,content-type');

使用动态scrpit标签,动态创建一个scrpit标签并指向请求的地址的方法,也就是JSONP方式,需要在URL后面拼接一个回调函数,标签加载成功以后会调用回调函数。var url = "http://uusama.com", callbaclName = 'jsonpCallback';

script = document.createElement('script');

script.type = 'text/javascript';

script.src = url + (url.indexOf('?') > -1 ? '&' : '?') + 'callback=' + callbaclName;

document.body.appendChild(script);

回调函数需要设置为全局函数:

window['jsonpCallback'] = function jsonpCallback(ret) {}

多个ajax请求数据同步问题

单个ajax返回数据异步处理

多个ajax请求互不相关,它们在被调用以后发送各自请求,请求成功以后调用自己的回调方法,互不影响。

因为ajax请求异步的特性,所有一些依赖于请求完成之后的操作我们都需要放在回调函数内部,否则的话,你在回调函数外面读取到的值是空。看下面的例子:var result = null;

ajax({

url: '/get.php?id=1',

method: 'GET',

success: function(ret) {

result = ret;

}

});

console.log(result); // 输出 null

虽然我们在回调函数里面设置了result的值,但是在最后一行 console.log(result); 输出为空。

因为ajax请求是异步的,程序执行到最后一行的时候,请求并没有完成,值并没有来得及修改。

这儿我们应该把 console.log(result) 相关的处理,放在 success 回调函数中才可以。

多个ajax返回数据问题

如果有多个ajax请求,情况会变得有些复杂。

如果多个ajax请求是按照顺序执行的,其中一个完成之后,才能进行下一个,则可以把后面一个请求放在前一后请求的回调中。

比如有两个ajax请求,其中一个请求的数据依赖于另外一个,则可以在第一个请求的回调里面再进行ajax请求:// 首先请求第一个ajax

ajax({

url: '/get1.php?id=1',

success: function(ret1) {

// 第一个请求成功回调以后,再请求第二个

if (ret1) {

ajax({

url: '/get2.php?id=4',

success:function(ret2) {

console.log(ret1);

console.log(ret2)

}

})

}

}

});

// 也可以写成下面的形式

// 将第二个ajax请求定义为一个函数,然后调用

var ajax2 = function(ret1) {

ajax({

url: '/get2.php?id=4',

success:function(ret2) {

console.log(ret1);

console.log(ret2)

}

});

};

ajax({

url: '/get1.php?id=1',

success: function(ret1) {

if(ret1){

ajax2(ret1); // 调用第二个ajax请求

}

}

});

如果不关心不同的ajax请求的顺序,而只是关心所有请求都完成,才能进行下一步。

一种方法是可以在每个请求完成以后都调用同一个回调函数,只有次数减少到0才执行下一步。var count = 3, all_ret = []; // 调用3次

ajax({

url: '/get1.php?id=1',

success:function(ret) {

callback(ret); // 请求成功后调用统一回调,次数减1

}

});

ajax({

url: '/get2.php?id=1',

success:function(ret) {

callback(ret);

}

});

ajax({

url: '/get3.php?id=1',

success:function(ret) {

callback(ret);

}

});

function callback(ret) {

// 当调用3次以上以后,说明3个ajax军完成

if (count > 0) {

count--; // 每调用一次,次数减1

// 可以在这儿保存 ret 到全局变量

all_ret.push(ret);

return;

} else { // 调用三次以后

// todo

console.log(ret);

}

}

另一种方法是设置一个定时任务去轮训是否所有ajax请求都完成,需要在每个ajax的成功回调中去设置一个标志。

这儿可以用是否获得值来判断,也可以设置标签来判断,用值来判断时,要注意设置的值和初始相同的情况。var all_ret = {

ret1: null, // 第一个ajax标识

ret2: null, // 第二个ajax标识

ret3: null, // 第三个ajax标识

};

ajax({

url: '/get1.php?id=1',

success:function(ret) {

all_ret['ret1'] = ret; // 修改第一个ajax请求标识

}

});

ajax({

url: '/get2.php?id=1',

success:function(ret) {

all_ret['ret2'] = ret; // 修改第二个ajax请求标识

}

});

ajax({

url: '/get3.php?id=1',

success:function(ret) {

all_ret['ret3'] = ret; // 修改第三个ajax请求标识

}

});

var repeat = setInterval(function(){

// 遍历是否所有ajax请求标识都已被修改,以此判断是否所有ajax请求都已完成

for(var item in all_ret) {

if (all_ret[item] === null){

return;

}

}

// todo, 到这儿所有ajax请求均已完成

clearInterval(repeat);

}, 50); // 调用次数可以适当调整,不应设的过小或者过大

以上就是本篇文章的所有内容,希望对大家学习提供到帮助!!

相关推荐:

php原生的异步请求,原生JavaScript实现Ajax异步请求相关推荐

  1. 获取referer中的请求参数_Servlet获取AJAX POST请求中参数以form data和request payload形式传输的方法...

    本文实例讲述了Servlet获取AJAX POST请求中参数以form data和request payload形式传输的方法.分享给大家供大家参考,具体如下: HTTP请求中,如果是get请求,那么 ...

  2. 微信ajax异步提交form表单,ajax异步提交表单

    前面我们一直谈 wojilu ajax framework ,已经表明了它是一个 ajax 框架.但其实客户端验证等内容几乎不涉及到 ajax 操作. 下面我们就着重看下这个框架ajax操作方面. 其 ...

  3. JavaScript小记——Ajax异步请求

    目录 Ajax介绍 Ajax使用 ES6拼串语法 Ajax介绍 Ajax    Asynchronous JavaScript and XML(异步的 JavaScript 和 XML). AJAX ...

  4. ajax异步提交局部刷新页面,ajax异步请求定时刷新局部页面

    用户提问 $(document).ready(function () { $.ajax({ type: 'GET', url:'http://localhost:8080/SSM/user/findR ...

  5. ajax 同步异步true,async: false 实现AJAX同步请求 ( $.ajax同步/异步(async:false/true) )(示例代码)...

    虽然说ajax用来执行异步请求的比较多,但有时还是存在需要同步执行的情况的. 比如:我需要通过ajax取执行请求以返回一个值,这个值在ajax后面是需要使用到的,这时就不能用异步请求了.这时候就需要使 ...

  6. ajax遍历数组对象数组对象数组,javascript - 从ajax json请求中,如何将对象动态添加到数组中,以便我可以遍历它们? - 堆栈内存溢出...

    您做对了,但是没有在正确的地方进行. jQuery.ajax不会等待响应,但会在请求得到响应时调用"成功"回调. 尝试这个: var i, venues, allVenues=[] ...

  7. ajax只请求一次,关于ajax的请求只处理一次的问题

    做登录ajax的时候, 发现同样的参数请求, 比如用户名zzz,密码zz, 我点击登录后:对于一个浏览器,你的第一次点击是会调用rpc请求的, 但是, 你再用同一个浏览器提交表单的时候, rpc不会被 ...

  8. ajax请求失败解决办法,Ajax跨越请求失败,解决

    跨越请求 1.1什么是跨域(两个不同系统之间的访问.调用) (1)域名不同,即两个不同的应用. (2)域名相同,但是端口不同,即同一个应用中的不同子系统. 1.2 Ajax跨域请求的缺陷 (1)创建t ...

  9. AJAX异步请求(Asynchronous Javascript And Xml)

    文章目录 1.传统请求及缺点 (1)传统的请求 (2)传统请求存在的问题 2.AJAX概述 3.XMLHttpRequest对象 4.AJAX GET请求 5.AJAX GET请求缓存问题 6.AJA ...

最新文章

  1. ERP 已死,中台已凉,DDD 称王!
  2. MySQL退出选择表格_MySQL常用命令
  3. 使用javadoc命令生成文档
  4. 5.创建表,使用alter进行表信息的增删改,Oracle回收站,集合运算
  5. 走近webpack(3)--图片的处理
  6. 爬取豆瓣top250的代码
  7. 8 使用SubMenu创建子菜单
  8. 步步为营,重构出模式(2)
  9. 大前端(全栈)学习路线指南
  10. [转]虚函数实现原理
  11. 较强冷空气影响中东部地区 江南华南等地有小到中雨
  12. java工作愿景_2018年年终总结以及来年目标愿景
  13. 计算机桌面机器人,桌面的 Spirit Rover(勇气号探测器),机器人爱好者的心头好!!...
  14. 四川理工计算机专业好不好,急!!!!四川理工 、 宜宾学院 、 内江师范学院 哪个学校计算机专业好点?...
  15. python运维是做什么的_Python运维工程师主要干什么?
  16. 「安全系列之CSRF」如何防范csrf攻击
  17. ubuntu下安装google拼音输入法
  18. php使用PHPexcel类读取excel文件(循环读取每个单元格的数据)
  19. 【c++】有理数加法
  20. Java编程之学生成绩

热门文章

  1. aspdotnet部分资源
  2. 利用IE浏览器进行web打印
  3. scrapy简单爬虫
  4. iOS--控制器加载自定义view的xib
  5. Ubuntu Linux环境下的Android开发环境的配置
  6. Visual SourceSafe使用流程指南
  7. 第五篇:数据预处理(二) - 异常值处理
  8. maven docker plugin 常见问题解决
  9. 【报告分享】人工智能新时代报告.pdf(附下载链接)
  10. Mahout的一些推荐算法