解决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 相关文章推荐
PHP 高手之路(一)
Oct 09 PHP
一些PHP写的小东西
Dec 06 PHP
Memcache 在PHP中的使用技巧
Feb 08 PHP
PHP--用万网的接口实现域名查询功能
Dec 13 PHP
ThinkPHP页面跳转success与error方法概述
Jun 25 PHP
PHP中is_file()函数使用指南
May 08 PHP
PHP aes (ecb)解密后乱码问题
Jun 22 PHP
php图片水印添加、压缩、剪切的封装类实现
Apr 18 PHP
PHP封装的MSSql操作类完整实例
May 26 PHP
laravel学习教程之关联模型
Jul 30 PHP
php实现有序数组旋转后寻找最小值方法
Sep 27 PHP
laravel-admin表单提交隐藏一些数据,回调时获取数据的方法
Oct 08 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
oracle资料库函式库
2006/10/09 PHP
原生JS可拖动弹窗效果实例代码
2013/11/09 Javascript
document节点对象的获取方式示例介绍
2013/12/24 Javascript
使用jQuery实现星级评分代码分享
2014/12/09 Javascript
javascript精确统计网站访问量实例代码
2015/12/19 Javascript
JQuery标签页效果实例详解
2015/12/24 Javascript
使用nodejs中httpProxy代理时候出现404异常的解决方法
2016/08/15 NodeJs
js 实现一些跨浏览器的事件方法详解及实例
2016/10/27 Javascript
基于bootstrap-datetimepicker.js不支持IE8的快速解决方法
2016/11/07 Javascript
原生js实现手风琴功能(支持横纵向调用)
2017/01/13 Javascript
jquery——九宫格大转盘抽奖实例
2017/01/16 Javascript
微信小程序通过api接口将json数据展现到小程序示例
2017/01/20 Javascript
JavaScript中捕获与冒泡详解及实例
2017/02/03 Javascript
利用forever和pm2部署node.js项目过程
2017/05/10 Javascript
vuex中使用对象展开运算符的示例
2017/09/25 Javascript
详解vuex 渐进式教程实例代码
2018/11/27 Javascript
一步一步实现Vue的响应式(对象观测)
2019/09/02 Javascript
layui: layer.open加载窗体时出现遮罩层的解决方法
2019/09/26 Javascript
JQuery样式与属性设置方法分析
2019/12/07 jQuery
JS时间戳与日期格式互相转换的简单方法示例
2021/01/30 Javascript
python 字符串转列表 list 出现\ufeff的解决方法
2017/06/22 Python
python自动发送邮件脚本
2018/06/20 Python
Django之Mode的外键自关联和引用未定义的Model方法
2018/12/15 Python
Django 实现xadmin后台菜单改为中文
2019/11/15 Python
使用pandas库对csv文件进行筛选保存
2020/05/25 Python
keras训练浅层卷积网络并保存和加载模型实例
2020/07/02 Python
HTML5响应式(自适应)网页设计的实现
2017/11/17 HTML / CSS
Smallable意大利家庭概念店:设计师童装及家居装饰
2018/01/08 全球购物
美国综合购物商城:UnbeatableSale.com
2018/11/28 全球购物
本科生职业生涯规划书范文
2014/01/21 职场文书
环境科学专业优秀毕业生自荐书
2014/02/03 职场文书
《鸟岛》教学反思
2014/04/26 职场文书
法学专业求职信
2014/07/15 职场文书
战马观后感
2015/06/08 职场文书
关于Vue Router的10条高级技巧总结
2021/05/06 Vue.js
详解MySQL中的pid与socket
2021/06/15 MySQL