解决AJAX请求使PHP反应时间过长的问题

现在我们开发了很多依赖于Ajax请求的应用,在某些情况下,甚至全部页面都依赖Ajax。有时我们会注意到,当一个网页发送两个或者更多Ajax请求时,PHP的反应时间会很长并会同时返回响应内容。

这个问题很可能是你处理PHP会话的方式造成的,跟着本文去理解这个问题,并做一些处理来避免这个问题。

内容

什么是PHP会话?
什么是Ajax?
具体问题
造成的原因
问题的解决方法
总结

什么是PHP会话?

为了理解这个问题,有必要先了解一下PHP会话和Ajax,以及它们是怎么干扰的。

假设你正在开发一个Web应用,想要识别不同用户。你想要记住是谁每次浏览了所有页面但都没有登录,这种情况下,你可以使用cookies或sessions。

可以你已经意识到了,sessions是一种存储用户信息的方式,它可以在任何页面重新取回用户信息。跟Cookies不同,Sessions是存储在服务器上的,所有用户不能直接改变这些信息。

默认情况下,Sessions在用户关闭浏览器之前均有效,或者用户在PHP配置文件中指定的时间内没有活动之后才失效。

在PHP页面中,无论你什么时候想要存储或者重新获取用户数据,都必须在页面开始处调用session_start(),因此才有权限使用$_SESSION去获取session数据。

什么是Ajax?

Ajax代表 Asynchronous JavaScript and XML,它是一种不用重新加载整个页面就能向服务器发送数据和从服务器接收数据的方式。

我们使用这种方式来发送数据,并以较快的速度从服务器检索数据。我们不用得到整个页面并在浏览器中渲染它,因为这是很慢的。

因此,我们可以更新网页的一部分,并且用户是可以看见这种改变的,就像用户向下滚动Facebook的时间线页面来看他们想看的东西,随着新内容添加而不用重载整个页面。

具体问题

开发几乎100%基于Ajax的Web应用已经不是一件新鲜事了,但是当一个网页同时发送两个及以上的Ajax请求时,你会注意到请求会花费很长时间,并且几乎在同一时刻完成了请求。

造成的原因

当你想服务器发送一个Ajax请求时,PHP脚本也开启了session_start(),它的调用会锁定PHP的session文件。

你可能已经知道,PHP默认会把session数据存储在服务器上的文件中。因为仅仅只有一个PHP请求能改变同一个session文件,两个同时的PHP请求可能会造成典型的文件锁条件,因此,任何一个其他由PHP调用的对于同一个用户的session_start()请求将不得不等到第一个请求结束。

现在,大部分PHP框架会首先在主文件中使用session_start()。因此,如果你正在使用会调用session_start()的框架或者函数库,将会造成session文件锁,对于使用同一个浏览器的相同用户,这将延迟同时发送的Ajax请求。

问题的解决方法

调用session_write_close()函数会使PHP写入session文件并关闭它,因此释放session文件后,另一个请求就有权限写入。

调用session_write_close()之后,当前脚本会继续正常运行,但你应该清楚在调用session_write_close()之后不允许改变任何session变量;在同一个脚本中,其它同时发送给PHP的请求可以锁定session文件并改变session变量。

为了让你看到这种问题,我创建了测试代码并将它上传到了github。你可以再这里找到测试脚本。在本地,你需要用一个实例来使用测试代码,然后打开浏览器控制台,查看请求和响应时间。

正如我们在这个文件中看到的示例代码,如果我们像下面代码这样,创建多个请求…

session_start();
sleep(5);

同一个用户的每个请求完成之前都将等到前一个请求完成。它将用时5s,因为知道脚本完成之前,session文件都没有被释放。因此,当第一次调用session_start()时,新的请求将被阻塞。那将杀死异步请求的想法,也就是说,多个请求会在同一时间被发送和被执行。

如果将文件中的代码改一下:

session_start();
// do something useful here
session_write_close();
sleep(5);

第三行代码将释放session文件锁,所以另一个并发请求不需要等待就能运行,因为它可以调用session_start()而不会有任何问题。

总结

PHP有些微妙,会让你担心为什么奇怪的事情会发生。但是一旦你理解事情是怎么运行的,一切将变得有意义,并且你能更好的思考去解决问题。

译文出处:http://www.ido321.com/1577.html

本文根据@Eslam Mahmoud的《Fix the AJAX Requests that Make PHP Take Too Long to Respond》所译,整个译文带有我自己的理解与思想,如果译得不好或有不对之处还请同行朋友指点。如需转载此译文,需注明英文出处:http://www.phpclasses.org/blog/post/277-Fix-the-AJAX-Requests-that-Make-PHP-Take-Too-Long-to-Respond.html

时间: 2024-08-01 06:31:09

解决AJAX请求使PHP反应时间过长的问题的相关文章

springMVC 解决ajax请求的406错误

直入主题-- 页面表单异步提交:$.post('<%=request.getContextPath()%>/web/register.html', $(form).serialize(), function(data){...}); java后台springMVC的controller接受请求: @RequestMapping("register") @ResponseBody public Map<String, String> register(HttpSe

解决ajax请求cors跨域问题

”已阻止跨源请求:同源策略禁止读取位于 ***** 的远程资源.(原因:CORS 头缺少 'Access-Control-Allow-Origin').“ ”已阻止跨源请求:同源策略禁止读取位于 ******的远程资源.(原因:CORS 请求失败).“ 在项目中或者练习中经常遇到ajax请求跨域的问题,除了可以用jsonp的请求模式,并且在后台支持回调的方式以外,还可以通过简单的配置webconfig文件或者IIS,解决该问题. 一.通过修改配置文件解决CORS跨域问题 在配置文件中的webse

解决ajax请求返回Json无法解析&quot;\&quot;字符的问题

原因:获取身份证信息,涉及图片路径,存在“\”字符,导致Json解析错误 解决思路:将返回类型从"json"改成"text",此时返回的数据类型变成字符串,将字符串的"\"替换成"/",此时不存在Json格式错误的问题, 再调用Json.parse解析成Json格式 $.ajax({ type : "GET", //提交方式 url : "http://127.0.0.1:24010/ZKIDRO

ajax请求后返回的时间转换格式

1.以前见过//@DateTimeFormat(pattern = "yyyy-MM-dd hh:mm:ss")  这种方式,但是最近用的时候不好使 2.一个比较不通用的方法解决了问题. import java.io.IOException; import java.text.SimpleDateFormat; import java.util.Date; import org.codehaus.jackson.JsonGenerator; import org.codehaus.ja

cors解决ajax请求跨域问题

Access-Control-Allow-Origin: * 适用tomcat部署的项目 在web.xml里添加以下内容 <filter> <filter-name>CorsFilter</filter-name> <filter-class>org.apache.catalina.filters.CorsFilter</filter-class> </filter> <filter-mapping> <filter

解决浏览器跨域限制发送ajax请求

一.什么是浏览器跨域限制?本质是什么? 所谓浏览器跨域限制,其实是为了数据安全的考虑由Netscape提出来限制浏览器跨域访问数据的策略,这是一中约定,正式叫法为浏览器同源策略,目前已经在大多数浏览器中支持. 本质上,所谓浏览器同源策略即:不允许浏览器访问跨域的Cookie,ajax请求跨域接口等.也就是说,凡是访问与自己不在相同域的数据或接口时,浏览器都是不允许的. 最常见的例子:对于前后端完全分离的Web项目,前端页面通过rest接口访问数据时,会出现如下问题: 不允许发送POST请求:在发

关于心跳ajax请求pending状态(被挂起),stalled时间过长的问题。涉及tcp连接异常。

环境:景安快云服务器(听说很垃圾,但是公司买的,我也刚来),CentOS-6.8-x86_64,Apache,MySQL5.1,PHP5.3. 问题:现公司有一个php系统,需要重复向后台发送ajax请求,但是会出现pending状态,我现在需要解决这个问题,或者说找到问题在服务器,代码,还是客户端,然后有个交代,但是不知道从何下手,毕竟还是it萌新啊.. 效果如图.两个特点,1:就是越往后的请求,pengding时间越长,且其中绝大部分时间被stalled占用(此问题网上有相关文章,但是没有解

ajax请求状态码为0的解决办法

原文链接:https://blog.csdn.net/changqing5818/article/details/53932463 前言 今天遇到个奇怪的问题,使用JQuery的ajax请求,后台的Controller层也接收到请求了,一切处理很正常,但是很奇怪,浏览器Debug发现,responseText是"error",状态码是0而不是200. 请求源码如下 $.ajax({ type: "POST", url: targetUrl, cache: false

客户端ajax请求为实现Token验证添加headers后导致正常请求变为options跨域请求解决方法

客户端为了实现token认证,通过Jquery的ajaxSetup方法全局配置headers: 全局配置headers后会导致部分不需要token认证的请求变为options请求,导致跨域访问.报错信息如下: CORS概念 支持CORS请求的浏览器一旦发现ajax请求跨域,会对请求做一些特殊处理,对于已经实现CORS接口的服务端,接受请求,并做出回应. 有一种情况比较特殊,如果我们发送的跨域请求为"非简单请求",浏览器会在发出此请求之前首先发送一个请求类型为OPTIONS的"