PHP的session文件锁引发的问题

1739次阅读 160人点赞 作者: WuBin 发布时间: 2025-02-19 16:17:03
扫码到手机查看

问题描述

最近做了个测试接口,专门用于测试主程序目录下接口的响应速度,目录结构如下:

pro:主程序目录
    |---server:用于存放各种后端服务
        |--web_api
            |-someApi
                |- common.php
            |-someApi
            |-...
            |-user_info.php
            |-_check_login.php
    |---index.html
    |--- ...
    |---api-test:测试程序
        |--someApi
        |--...
        |-- test_common.php

主程序中的代码

在web_api/someApi/common.php 中引入了 外层的user_info个文件:

include_once (dirname(__DIR__) . '/user_info.php');

然后user_info则引入了 ——_check_login.php

include_once (dirname(__DIR__) . '/server/_check_login.php');

// session_start() 写在这里也不行!!

_check_login.php的代码如下所示:

<?php
session_start();

if ( $isLogin ) {
  $resultLogin = array(
     ...
  );
  echo json_encode($resultLogin, JSON_UNESCAPED_UNICODE);
  exit;
} else {
  ...
}
?>

测试程序中的代码

再来看测试程序中的代码,api-test/test_common.php:

<?php
ini_set('date.timezone','Asia/Shanghai');
ini_set('max_execution_time', '120');
set_time_limit(120);
sleep(10);
// 启动session会话
session_start();

问题描述

然后,我先用浏览器访问api-test/someApi/app.php的时候,由于添加了sleep,会发现请求一直在处理中。此时,我再用浏览器打开pro/index.html,会发现在pro/index.html中的请求也一同处于等待!如下图:

问题解答

导致这个问题出现的根本原因,就是session_start!!

会话文件锁定机制

PHP 默认使用文件来存储会话数据,当调用session_start()时,PHP 会尝试对会话文件加锁,以确保同一时间只有一个请求可以修改该会话数据。如果多个请求尝试同时访问同一个会话,后面的请求会被阻塞,直到前面的请求释放锁。

假设有一个 PHP 页面test.php

<?php
session_start();
// 模拟一个耗时操作
sleep(10);
echo "Process finished.";
?>

当你同时打开两个浏览器窗口访问test.php时,第二个请求会被阻塞,直到第一个请求执行完毕并释放会话锁。

所以最终问题原因就是api-test/test_common.php中的session_start()!!把这里的session_start去掉就一切正常了!!

当然也可以用其他方式解决这个问题:

缩短会话锁定时间

尽量减少在会话锁定期间执行的耗时操作。例如,将一些不必要的操作放在

尽量减少在会话锁定期间执行的耗时操作。例如,将一些不必要的操作放在session_start()之前或之后执行。

// 执行一些不依赖会话的操作
// ...

session_start();
// 执行依赖会话的操作
// ...

// 关闭会话,释放锁
session_write_close();

// 继续执行其他操作
// ...

使用数据库存储会话数据

将会话数据存储在数据库中,而不是文件系统中,可以避免文件锁定带来的问题。你可以通过自定义会话处理函数来实现,示例代码如下:
// 自定义会话处理函数
class DatabaseSessionHandler implements SessionHandlerInterface {
    // 实现接口中的方法
    // ...
}

// 注册自定义会话处理函数
$handler = new DatabaseSessionHandler();
session_set_save_handler($handler, true);

// 启动会话
session_start();

资源竞争问题

如果服务器资源有限,例如磁盘 I/O、内存等,多个请求同时调用session_start()可能会导致资源竞争,从而出现阻塞现象。

  • 优化服务器配置:增加服务器的硬件资源,如增加内存、更换高速磁盘等,以提高服务器的处理能力。
  • 使用缓存:对于一些频繁访问的数据,可以使用缓存技术(如 Redis、Memcached)来减少对会话文件或数据库的访问,从而减轻服务器的负担。
点赞 支持一下 觉得不错?客官您就稍微鼓励一下吧!
关键词:session_start,文件锁
推荐阅读
  • uniapp实现被浏览器唤起的功能

    当用户打开h5链接时候,点击打开app若用户在已经安装过app的情况下直接打开app,若未安装过跳到应用市场下载安装这个功能在实现上主要分为两种场景,从普通浏览器唤醒以及从微信唤醒。

    10350次阅读 683人点赞 发布时间: 2022-12-14 16:34:53 立即查看
  • Vue

    盘点Vue2和Vue3的10种组件通信方式

    Vue中组件通信方式有很多,其中Vue2和Vue3实现起来也会有很多差异;本文将通过选项式API组合式API以及setup三种不同实现方式全面介绍Vue2和Vue3的组件通信方式。

    4859次阅读 365人点赞 发布时间: 2022-08-19 09:40:16 立即查看
  • JS

    几个高级前端常用的API

    推荐4个前端开发中常用的高端API,分别是MutationObserver、IntersectionObserver、getComputedstyle、getBoundingClientRect、requ...

    14957次阅读 987人点赞 发布时间: 2021-11-11 09:39:54 立即查看
  • PHP

    【正则】一些常用的正则表达式总结

    在日常开发中,正则表达式是非常有用的,正则表达式在每个语言中都是可以使用的,他就跟JSON一样,是通用的。了解一些常用的正则表达式,能大大提高你的工作效率。

    14220次阅读 548人点赞 发布时间: 2021-10-09 15:58:58 立即查看
  • 【中文】免费可商用字体下载与考证

    65款免费、可商用、无任何限制中文字体打包下载,这些字体都是经过长期验证,经得住市场考验的,让您规避被无良厂商起诉的风险。

    13040次阅读 1055人点赞 发布时间: 2021-07-05 15:28:45 立即查看
  • Vue

    Vue3开发一个v-loading的自定义指令

    在vue3中实现一个自定义的指令,有助于我们简化开发,简化复用,通过一个指令的调用即可实现一些可高度复用的交互。

    17209次阅读 1382人点赞 发布时间: 2021-07-02 15:58:35 立即查看
  • JS

    关于手机上滚动穿透问题的解决

    当页面出现浮层的时候,滑动浮层的内容,正常情况下预期应该是浮层下边的内容不会滚动;然而事实并非如此。在PC上使用css即可解决,但是在手机端,情况就变的比较复杂,就需要禁止触摸事件才可以。

    15655次阅读 1269人点赞 发布时间: 2021-05-31 09:25:50 立即查看
  • Vue

    Vue+html2canvas截图空白的问题

    在使用vue做信网单页专题时,有海报生成的功能,这里推荐2个插件:一个是html2canvas,构造好DOM然后转canvas进行截图;另外使用vue-canvas-poster(这个截止到2021年3月...

    30981次阅读 2437人点赞 发布时间: 2021-03-02 09:04:51 立即查看
  • Vue

    vue-router4过度动画无效解决方案

    在初次使用vue3+vue-router4时候,先后遇到了过度动画transition进入和退出分别无效的情况,搜遍百度没没找到合适解决方法,包括vue-route4有一些API都进行了变化,以前的一些操...

    26813次阅读 2062人点赞 发布时间: 2021-02-23 13:37:20 立即查看
交流 收藏 目录