解决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 页面跳转到另一个页面的多种方法方法总结
Jul 07 PHP
php 验证码制作(网树注释思想)
Jul 20 PHP
php 输出双引号"与单引号'的方法
May 09 PHP
解析PHP工厂模式的好处
Jun 18 PHP
将word转化为swf 如同百度文库般阅读实现思路及代码
Aug 09 PHP
php读取csv数据保存到数组的方法
Jan 03 PHP
php文件下载处理方法分析
Apr 22 PHP
php 无限级分类 获取顶级分类ID
Mar 13 PHP
php实现简单加入购物车功能
Mar 07 PHP
php获取ajax的headers方法与内容实例
Dec 27 PHP
php探针使用原理和技巧讲解
Sep 17 PHP
PHP7 字符串处理机制修改
Mar 09 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基础知识:类与对象(5) static
2006/12/13 PHP
php不使用copy()函数复制文件的方法
2015/03/13 PHP
自己做的模拟模态对话框实现代码
2012/05/23 Javascript
简单的Jquery全选功能
2013/11/07 Javascript
推荐一款jQuery插件模板
2015/01/09 Javascript
JQuery中clone方法复制节点
2015/05/18 Javascript
简单讲解jQuery中的子元素过滤选择器
2016/04/18 Javascript
在 Typescript 中使用可被复用的 Vue Mixin功能
2018/04/17 Javascript
解决vue中post方式提交数据后台无法接收的问题
2018/08/11 Javascript
vue-image-crop基于Vue的移动端图片裁剪组件示例
2018/08/28 Javascript
koa中间件核心(koa-compose)源码解读分析
2020/06/15 Javascript
js实现扫雷源代码
2020/11/27 Javascript
[02:18]DOTA2英雄基础教程 育母蜘蛛
2014/01/20 DOTA
[15:56]Heroes18_暗影萨满(完美)
2014/10/31 DOTA
Python利用Nagios增加微信报警通知的功能
2016/02/18 Python
Python中使用haystack实现django全文检索搜索引擎功能
2017/08/26 Python
python中的decimal类型转换实例详解
2019/06/26 Python
Python3.7 读取音频根据文件名生成脚本的代码
2020/04/07 Python
Python Pygame实现俄罗斯方块
2021/02/19 Python
python 实现有道翻译功能
2021/02/26 Python
Lookfantastic葡萄牙官方网站:欧洲第一大化妆品零售商
2018/03/17 全球购物
adidas澳大利亚官方网站:adidas Australia
2018/04/15 全球购物
SIDESTEP荷兰:在线购买鞋子
2019/11/18 全球购物
PHP如何删除一个Cookie值
2012/11/15 面试题
《雨点》教学反思
2014/02/12 职场文书
yy司仪主持词
2014/03/22 职场文书
小学六年级学生评语
2014/04/22 职场文书
小学生综合素质评语
2014/04/23 职场文书
趣味运动会策划方案
2014/06/02 职场文书
法学自荐信
2014/06/20 职场文书
大学生学习面向未来的赶考思想汇报
2014/09/12 职场文书
办理信用卡收入证明范例
2014/09/13 职场文书
党员批评与自我批评发言
2014/10/02 职场文书
销售经理岗位职责范本
2015/04/02 职场文书
社区环境卫生倡议书
2015/04/29 职场文书
使用logback实现按自己的需求打印日志到自定义的文件里
2021/08/30 Java/Android