浅析PHP中Session可能会引起并发问题


Posted in PHP onJuly 23, 2015

在进行Web应用程序开发的时候,人们经常会用Session存储数据。但可能有人不知道,在PHP中,Session使用不当可能会引起并发问题。印度医疗行业软件解决方案提供商Plus91 Technologies高级工程师Kishan Gor在个人博客上对这个问题进行了阐释。

如果同一个客户端并发发送多个请求,而每个请求都使用了Session,那么PHP Session锁的存在会导致服务器串行响应这些请求,而不是并行。这是因为在默认情况下,PHP使用文件存储Session数据。对于每一个新的Session,PHP会创建一个文件,并持续向其中写入数据。所以,每次调用session_start()方法,就会打开Session文件,并取得文件的独占锁。这样,如果服务器脚本正在处理一个请求,而客户端又发送了一个同样需要使用Session的请求,那么后一个请求会阻塞,直至前一个请求处理完成释放了文件上的独占锁。不过,这只限于来自同一个客户端的多个请求,也就是说,来自一个客户端的请求并不会阻塞另一个客户端的请求。

如果脚本很短,这通常没有问题。但如果脚本运行时间比较长,那就可能会产生问题。在现代Web应用程序开发中,有一个非常常见的情况,就是使用AJAX技术在同一个页面内发送多个请求获取数据。如果这些请求都需要使用Session,那么第一个请求到达服务器后会取得Session锁,其它请求就必须等待,所有请求将串行处理,即使它们彼此之间并没有依赖关系。这将大大增加页面的响应时间。

有一个方法可以避免这个问题,就是在使用完Session以后立即调用session_write_close()方法关闭Session。这样Session锁就会释放,即使当前脚本还在等在处理。需要注意的是,调用该方法后,当前脚本就不能进一步操作Session了。

需要特别指出的是,本文所陈述的问题和观点只适用于使用session_start()方法的PHP默认Session管理模式。比如,有用户就指出,如果将应用程序托管在AWS EC2上,并正确配置DynamoDB,Session锁定问题就不会出现。

附上一份实例代码:

Session.php

<?php

final class SessionController extends YafController_Abstract
{
  public function setUserFileAction()
  {
    session_start();
    $_SESSION['user_name'] = 'xudianyang';
    $_SESSION['user_id']  = '123';

    sleep(3);
    echo json_encode($_SESSION);
    return false;
  }

  public function setLoginFileAction()
  {
    session_start();
    $_SESSION['last_time'] = time();

    echo json_encode($_SESSION);
    return false;
  }

  public function indexFileAction()
  {
    // Auto Rend View
  }

  public function getSessionFileAction()
  {
    session_start();
    var_dump($_SESSION);

    return false;
  }

  public function setUserRedisAction()
  {
    $session = CoreFactory::session();
    $session->set('user_name', 'xudianyang');
    $session->set('user_id', '123');

    sleep(3);
    echo json_encode($_SESSION);
    return false;
  }

  public function setLoginRedisAction()
  {
    $session = CoreFactory::session();
    $session->set('last_time', time());

    echo json_encode($_SESSION);
    return false;
  }

  public function indexRedisAction()
  {
    // Auto Rend View
  }

  public function getSessionRedisAction()
  {
    $session = CoreFactory::session();
    var_dump($_SESSION);

    return false;
  }
}

indexfile.phtml

<!DOCTYPE html>
<html>
<head>
 <title>测试session并发锁问题</title>
 <meta charset="utf-8">
 <script type="text/javascript" src="/assets/js/jquery-1.10.2.min.js"></script>
 <script type="text/javascript">
   $.ajax({
     url: "/session/setUserFile",
     type: "get",
     dataType: "json",
     success: function(response){
       console.info(response.last_time);
     }
   });
   setTimeout(function(){
     $.ajax({
       url: "/session/setLoginFile",
       type: "get",
       dataType: "json",
       success: function(response){
         console.info(response.last_time);
       }
     });
   }, 300);
 </script>
</head>
<body>
同时发起2两个ajax请求
</body>
</html>

indexredis.phtml

<!DOCTYPE html>
<html>
<head>
 <title>测试session并发锁问题</title>
 <meta charset="utf-8">
 <script type="text/javascript" src="/assets/js/jquery-1.10.2.min.js"></script>
 <script type="text/javascript">
   $.ajax({
     url: "/session/setUserRedis",
     type: "get",
     dataType: "json",
     success: function(response){
       console.info(response.last_time);
     }
   });
   setTimeout(function(){
     $.ajax({
       url: "/session/setLoginRedis",
       type: "get",
       dataType: "json",
       success: function(response){
         console.info(response.last_time);
       }
     });
   }, 300);
 </script>
</head>
<body>
同时发起2两个ajax请求
</body>
</html>

以上所述就是本文的全部内容了,希望大家能够喜欢。

PHP 相关文章推荐
Discuz 5.0 中读取纯真IP数据库函数分析
Mar 16 PHP
php Http_Template_IT类库进行模板替换
Mar 19 PHP
php+mysql事务rollback&amp;commit示例
Feb 08 PHP
用PHP获取Google AJAX Search API 数据的代码
Mar 12 PHP
PHP 文件缓存的性能测试
Apr 25 PHP
php 记录进行累加并显示总时长为秒的结果
Nov 04 PHP
PHP实现图片压缩的两则实例
Jul 19 PHP
PHP获取Exif缩略图的方法
Jul 13 PHP
php使用json_decode后数字对象转换成了科学计数法的解决方法
Feb 20 PHP
yii2.0整合阿里云oss删除单个文件的方法
Sep 19 PHP
通过PHP实现获取访问用户IP
May 09 PHP
PHP7 弃用功能
Mar 09 PHP
PHP技术开发微信公众平台
Jul 22 #PHP
PHP使用array_merge重新排列数组下标的方法
Jul 22 #PHP
PHP结合jQuery实现找回密码
Jul 22 #PHP
使用PHP生成二维码的方法汇总
Jul 22 #PHP
使用PHP编写发红包程序
Jul 22 #PHP
解决nginx不支持thinkphp中pathinfo的问题
Jul 21 #PHP
php 把数字转换成汉字的代码
Jul 21 #PHP
You might like
PHP中的integer类型使用分析
2010/07/27 PHP
学习php设计模式 php实现合成模式(composite)
2015/12/08 PHP
用JS实现的一个include函数
2007/07/21 Javascript
js对象关系图 方便dom操作
2012/03/18 Javascript
JavaScript判断密码强度(自写代码)
2013/09/06 Javascript
获取select元素被选中的文本内容的js代码
2014/01/29 Javascript
jquery实现带二级菜单的导航示例
2014/04/28 Javascript
JQuery Tips相关(1)----关于$.Ready()
2014/08/14 Javascript
js在数组中删除重复的元素自保留一个(两种实现思路)
2014/08/22 Javascript
jquery分割字符串的方法
2015/06/24 Javascript
javascript实现点击提交按钮后显示loading的方法
2015/07/03 Javascript
JavaScript实现的经典文件树菜单效果
2015/09/08 Javascript
javascript中错误使用var造成undefined
2016/03/31 Javascript
JS获取屏幕高度的简单实现代码
2016/05/24 Javascript
JS实现刷新父页面不弹出提示框的方法
2016/06/22 Javascript
jQuery Validate插件自定义验证规则的方法
2016/12/27 Javascript
Vue.js仿Metronic高级表格(二)数据渲染
2017/04/19 Javascript
webpack4 从零学习常用配置(小结)
2019/05/28 Javascript
layui复选框限制选择个数的方法
2019/09/18 Javascript
layui 数据表格 根据值(1=业务,2=机构)显示中文名称示例
2019/10/26 Javascript
Vue-CLI 3 scp2自动部署项目至服务器的方法
2020/07/24 Javascript
[56:41]2018DOTA2亚洲邀请赛 3.31 小组赛 A组 Newbee vs OG
2018/04/01 DOTA
[39:21]LGD vs OG 2019国际邀请赛淘汰赛 胜者组 BO3 第二场 8.24
2019/09/10 DOTA
Python一句代码实现找出所有水仙花数的方法
2018/11/13 Python
opencv3/python 鼠标响应操作详解
2019/12/11 Python
python实现上传文件到linux指定目录的方法
2020/01/03 Python
纯CSS3制作的鼠标悬停时边框旋转
2017/01/03 HTML / CSS
英国家电购物网站:Sonic Direct
2019/03/26 全球购物
英国奢侈品概念店:Base Blu
2019/05/16 全球购物
ORACLE第二个十问
2013/12/14 面试题
班长自荐书范文
2014/02/11 职场文书
我们的节日清明节活动方案
2014/03/05 职场文书
老干部工作汇报材料
2014/10/28 职场文书
初三毕业评语
2014/12/26 职场文书
出租车拒载检讨书
2015/01/28 职场文书
MySQL开启事务的方式
2021/06/26 MySQL