解决PHP Opcache 缓存刷新、代码重载出现无法更新代码的问题


Posted in PHP onAugust 24, 2020

问题背景

通过启用Opcache的缓存优化,将PHP代码预编译为Opcode缓存到共享内存中供进程反复调用,从而减少了重复从磁盘解析PHP代码的时间消耗,显著的提高了PHP性能,提升了业务性能的调用,但是也引发了一些问题,就是我们每次更新了相应的PHP代码后,web server 无法即时加载到更新后的代码。

解决方案

(一)、设置Opcache脚本验证时间

可以通过更改 Opcache 以下两个配置选项来调整代码重载时间

opcache.revalidate_freq=0 检查脚本时间戳是否有更新的周期,以秒为单位。(如果设置为 0 会导致针对每个请求, OPcache 都会检查脚本更新)

opcache.validate_timestamps=0 如果启用,那么 OPcache 会每隔 opcache.revalidate_freq 设定的秒数 检查脚本是否更新。

PS:在实际生产环境中,为了尽可能达到最优性能,尽量不开启文件更新验证,因为每次验证都会重新预编译PHP代码到共享内存中。

(二)、重启 | 重载 php-fpm 进程

每次重启或重启 php-fpm 进程便会重新解析PHP脚本文件,但是重启 fpm 进程可能会导致请求中断,从而导致写入脏数据 或者 造成事务回滚等一系列异常。

重载相对于重启则平顺很多,不会导致用户请求直接中断,相对来说风险低很多,但是php-fpm 收到reload信号,便会向所有子进程发送SIGGUIT信号,同时注册一个定时器,在规定的时间之内子进程没有退出,接着在发送SIGTERM信号,结束子进程。如果在一秒之内子进程还是没结束 直接发送SIGKILL 强制杀死。

重启php-fpm

service php-fpm restart

重载php-fpm

services php-fpm reload

或 kill -USR2 `cat /usr/local/php/var/run/php-fpm.pid`

(三)、手动清理缓存

除了上面的两种方式,还有更为稳妥一点的缓存清理,我们可以通过opcache_reset()和opcache_invalidate() 函数来刷新Opcache缓存。

opcache_reset() - 重置整个Opcode缓存,所有的PHP脚本将会被重新解析再预编译为Opcode。

opcache_invalidate() - 清除指定脚本缓存,可以传递两个参数,一个是刷新文件路径,一个是force字段, 如果 force 没有设置或者传入的是 FALSE,那么只有当脚本的修改时间 比对应Opcode的时间更新时,脚本的缓存才会失效。

需要注意的是,当PHP以PHP-FPM的方式运行的时候,opcache的缓存是无法通过php命令进行清除的,只能通过http或cgi到php-fpm进程的方式来清除缓存,我们可以编写一个对外接口,来达到清理缓存的目的。

相关实现如下(框架:laravel):

Route::any('cache-reset', function () {
  //重置整个Opcode缓存
  dd(opcache_reset());
});

Route::any('cache-update', function () {
  //清除掉最近一次更新文件的缓存
  exec('git diff --name-only HEAD~ HEAD', $output);
  foreach ($output as $file) {
    $path = base_path($file);
    opcache_invalidate($path, true);
  }
  dd('刷新完成');
});

总结

通过上面的三种策略,可以实现 Opcache 缓存更新的目的,但是在流量高峰期或者大流量的服务端,每次更新缓存都是一件非常损耗资源的事情,Opcache在重建缓存时,也不会禁止其他进程读取,因此就会造成反复新建缓存,因此想要达到最佳的性能调配:

  • 最好不要在高峰期清理缓存
  • 高峰期不要频繁的更新代码,清理缓存,会造成重复新建缓存
  • 如果需要更新,可以尝试削弱服务端权重,实现逐个更新的目的。
  • 如果需要强制更新,尽量选择手动清除缓存的方式,来重建Opcache缓存,使代价最小化。

以上就是解决PHP Opcache 缓存刷新、代码重载出现无法更新代码的问题的详细内容,更多关于PHP Opcache 缓存刷新、代码重载的资料请关注三水点靠木其它相关文章!

PHP 相关文章推荐
BBS(php & mysql)完整版(八)
Oct 09 PHP
谈谈PHP的输入输出流
Feb 14 PHP
php面向对象全攻略 (十) final static const关键字的使用
Sep 30 PHP
php 上传文件类型判断函数(避免上传漏洞 )
Jun 08 PHP
PHP 5.5 创建和验证哈希最简单的方法详解
Nov 07 PHP
phpQuery占用内存过多的处理方法
Nov 13 PHP
ThinkPHP3.1新特性之Action参数绑定
Jun 19 PHP
PHP判断字符串长度的两种方法很实用
Sep 22 PHP
PHP is_array() 检测变量是否是数组的实现方法
Jun 13 PHP
PHP封装的page分页类定义与用法完整示例
Dec 24 PHP
Laravel 类和接口注入相关的代码
Oct 15 PHP
PHP执行linux命令6个函数代码实例
Nov 24 PHP
WordPress免插件实现面包屑导航的示例代码
Aug 20 #PHP
VSCode+PHPstudy配置PHP开发环境的步骤详解
Aug 20 #PHP
Laravel相关的一些故障解决
Aug 19 #PHP
聊聊 PHP 8 新特性 Attributes
Aug 19 #PHP
php提高脚本性能的4个技巧
Aug 18 #PHP
php判断IP地址是否在多个IP段内
Aug 18 #PHP
PHP 8新特性简介
Aug 18 #PHP
You might like
解析PHP中如何将数组变量写入文件
2013/06/06 PHP
四个常见html网页乱码问题及解决办法
2015/09/08 PHP
一个挺有意思的Javascript小问题说明
2011/09/26 Javascript
JS.findElementById()使用介绍
2013/09/21 Javascript
jQuery 文本框得失焦点的简单实例
2014/02/19 Javascript
jQuery实现企业网站横幅焦点图切换功能实例
2015/04/30 Javascript
jQuery实现div拖拽效果实例分析
2016/02/20 Javascript
使用vue.js开发时一些注意事项
2016/04/27 Javascript
AnjularJS中$scope和$rootScope的区别小结
2016/09/18 Javascript
微信小程序利用swiper+css实现购物车商品删除功能
2019/03/06 Javascript
JavaScript实现美化滑块效果
2019/05/17 Javascript
JavaScript计算出两个数的差值
2020/03/19 Javascript
vue+Element-ui实现分页效果
2020/11/15 Javascript
linux系统使用python获取内存使用信息脚本分享
2014/01/15 Python
零基础写python爬虫之爬虫编写全记录
2014/11/06 Python
Python下的twisted框架入门指引
2015/04/15 Python
从Python程序中访问Java类的简单示例
2015/04/20 Python
Python解析excel文件存入sqlite数据库的方法
2016/11/15 Python
Python基于正则表达式实现检查文件内容的方法【文件检索】
2017/08/30 Python
Python使用requests发送POST请求实例代码
2018/01/25 Python
使用Python爬了4400条淘宝商品数据,竟发现了这些“潜规则”
2018/03/23 Python
Python读取本地文件并解析网页元素的方法
2018/05/21 Python
Windows下PyCharm安装图文教程
2018/08/27 Python
python中正则表达式与模式匹配
2019/05/07 Python
Python自动化运维之Ansible定义主机与组规则操作详解
2019/06/13 Python
Python退出时强制运行一段代码的实现方法
2020/04/29 Python
python pip如何手动安装二进制包
2020/09/30 Python
利用python+request通过接口实现人员通行记录上传功能
2021/01/13 Python
html5+css3气泡组件的实现
2014/11/21 HTML / CSS
后勤人员自我评价怎么写
2013/09/19 职场文书
组织关系转移介绍信
2014/01/16 职场文书
2014教师评职称工作总结
2014/11/10 职场文书
清明节扫墓活动总结
2015/02/09 职场文书
django注册用邮箱发送验证码的实现
2021/04/18 Python
python实现三次密码验证的示例
2021/04/29 Python
VMware虚拟机安装 Windows Server 2022的详细图文教程
2022/09/23 Servers