在 Docker 中使用 OPcache 加速 PHP

发布日期:作者:

Speeding Up PHP with OPcache in Docker image

如果您使用的是 Docker for Mac 或 Docker for Windows,您可能会注意到一些明显的延迟和首字节时间 (TTFB),具体取决于您的应用程序的设置。提高性能最重要的事情之一是启用 OPCache 模块(无论开发环境如何)。还有其他一些方法,例如卷缓存(如果可能),但 OPcache 是您在运行 PHP 应用程序的 **任何** 环境中都想要获得的优势。

OPcache 设置

启用 OPCache 模块后,您需要考虑一些事项,以确保您的配置适合开发,同时如果计划在生产环境中使用 Docker,则可以为生产环境做好准备。

以下是您在开发过程中最终获得的粗略配置

[opcache]
opcache.enable=1
; 0 means it will check on every request
; 0 is irrelevant if opcache.validate_timestamps=0 which is desirable in production
opcache.revalidate_freq=0
opcache.validate_timestamps=1
opcache.max_accelerated_files=10000
opcache.memory_consumption=192
opcache.max_wasted_percentage=10
opcache.interned_strings_buffer=16
opcache.fast_shutdown=1

请注意,我们已对这些值进行了硬编码,这在不同环境之间缺乏灵活性。我们稍后会回来使其更灵活!

开发中最重要的一项设置是 opcache.validate_timestamps=1,它允许我们对代码进行更改。如果您使用的是 Docker 卷,这意味着 OPcache 会尊重文件时间戳,您的更改将立即反映出来。在生产环境中,这并不理想,这就是我们的动态配置稍后将发挥作用的地方。

您不应在不了解它们的功能的情况下直接复制/粘贴这些设置。该配置主要来自 Steve Corona 的文章 最佳 Zend OpCache 设置/调整/配置,这是一份了解这些值含义的优秀资源。有关性能(包括 OPcache)的另一个优秀资源是 Chris Fidao 的 扩展 Laravel

在 Dockerfile 中复制 INI 设置

以下是用于安装 OPcache 模块并在其中复制 INI 文件以配置 OPCache 的粗略 Dockerfile

FROM php:7.2-apache-stretch
 
RUN docker-php-ext-install opcache
 
COPY docker/php/conf.d/opcache.ini /usr/local/etc/php/conf.d/opcache.ini
COPY . /var/www/html

我没有展示完整的 PHP 应用程序工作示例。Laravel 的文档根目录需要是 /var/www/html/public,但我只是试图在这篇文章中演示如何设置 OPcache。有关完整示例,您应查看我的课程 Docker for PHP 开发人员

Dockerfile 假设以下文件夹结构用于组织您的 Docker 文件

├── app
├── bootstrap
├── config
├── database
├── docker
│   └── php
│   └── conf.d
├── public
├── resources
├── routes
├── storage
├── tests
└── vendor

请注意,在实际项目中,您可能会使用一个基础镜像,而不是将所有这些内容放在您的项目中,但我在这里展示了这些内容,以便您可以遵循 OPcache 的特定配置。

构建 Dockerfile

以下是可以运行的构建命令,用于尝试配置 OPcache

docker build --pull -t opcache-demo -f docker/Dockerfile .
docker run --rm -it opcache-demo bash
# In a running container:
/var/www/html# php -m | grep OPcache
Zend OPcache

使用环境进行灵活配置

我们已启用 OPcache,但如果我们想使此配置更灵活,可以使用环境变量来配置 INI 设置

[opcache]
 
opcache.enable=1
opcache.revalidate_freq=0
opcache.validate_timestamps=${PHP_OPCACHE_VALIDATE_TIMESTAMPS}
opcache.max_accelerated_files=${PHP_OPCACHE_MAX_ACCELERATED_FILES}
opcache.memory_consumption=${PHP_OPCACHE_MEMORY_CONSUMPTION}
opcache.max_wasted_percentage=${PHP_OPCACHE_MAX_WASTED_PERCENTAGE}
opcache.interned_strings_buffer=16
 
opcache.fast_shutdown=1

现在我们有了环境驱动的 INI 文件,让我们在 Dockerfile 中为我们的项目提供一些默认值

FROM php:7.2-apache-stretch
 
ENV PHP_OPCACHE_VALIDATE_TIMESTAMPS="0" \
PHP_OPCACHE_MAX_ACCELERATED_FILES="10000" \
PHP_OPCACHE_MEMORY_CONSUMPTION="192" \
PHP_OPCACHE_MAX_WASTED_PERCENTAGE="10"
 
RUN docker-php-ext-install opcache
 
COPY docker/php/conf.d/opcache.ini /usr/local/etc/php/conf.d/opcache.ini
COPY . /var/www/html

请注意,默认情况下,我们将禁用时间戳,因此我们需要在开发中覆盖此环境值。在这篇文章中,我们将不讨论使用 Docker Compose 设置环境,但以下是您可以运行的粗略命令,以确保在开发过程中验证时间戳

# Rebuild the image first
docker build --pull -t opcache-demo -f docker/Dockerfile .
 
docker run --rm -d \
-p 8080:80 \
-e "PHP_OPCACHE_VALIDATE_TIMESTAMPS=1" \
opcache-demo

在 Apache 容器在后台运行的情况下,您可以通过在容器中验证来确认 OPcache 时间戳设置是否为 1

# get the container id
docker ps
docker exec -it 6002d83c6d24 bash
 
# In a running container:
/var/www/html# php -i | grep validate_timestamps
opcache.validate_timestamps => On => On

如您所见,我们的配置现在由环境变量动态驱动!在 Docker 中,您的代码默认情况下将使用 OPCache 进行缓存,并且不会由于时间戳验证被禁用而更新。请注意,如果您使用的是 Nginx + PHP-FPM,您需要确保您的 FPM 池(可能是 www)中包含 clear_env = no

[www]
clear_env = no

您也可以手动将环境变量添加到池中,如果您不想将整个环境提供给 PHP。

了解更多

虽然本文中介绍的思路并非 Docker 独有,但 OPcache 在开发中提供的额外帮助非常有用,而且不会影响您更新代码的能力。

如果您想了解更多关于使用 Docker 和 PHP(包括 Laravel)开发 PHP 应用程序的信息,请查看我的课程 Docker for PHP 开发人员


包含的链接是联盟链接,这意味着如果您决定购买,Laravel News 会获得一小部分收益,以帮助运行此网站。

Paul Redmond photo

Laravel News 的工作人员。全栈 Web 开发人员和作者。

Cube

Laravel 新闻通讯

加入 40,000 多名其他开发人员,永不错过新的技巧、教程等。

Laravel Forge logo

Laravel Forge

轻松创建和管理您的服务器,并在几秒钟内部署您的 Laravel 应用程序。

Laravel Forge
Tinkerwell logo

Tinkerwell

Laravel 开发人员必备的代码运行器。使用 AI、自动完成和本地和生产环境中的即时反馈进行调试。

Tinkerwell
No Compromises logo

无妥协

Joel 和 Aaron,来自 No Compromises 播客的两名经验丰富的开发者,现在可供您为您的 Laravel 项目聘用。 ⬧ 固定费率为 7,500 美元/月。 ⬧ 没有冗长的销售流程。 ⬧ 没有合同。 ⬧ 100% 返款保证。

无妥协
Kirschbaum logo

Kirschbaum

提供创新和稳定性,以确保您的 Web 应用程序取得成功。

Kirschbaum
Shift logo

Shift

运行旧版本的 Laravel?即时、自动化的 Laravel 升级和代码现代化,让您的应用程序保持新鲜。

Shift
Bacancy logo

Bacancy

使用经验丰富的 Laravel 开发人员(4-6 年经验)以每月仅 2,500 美元的价格为您的项目增效。获得 160 小时的专业服务和 15 天无风险试用。立即预约电话!

Bacancy
Lucky Media logo

Lucky Media

立即获得 Lucky - Laravel 开发的理想选择,拥有超过十年的经验!

Lucky Media
Lunar: Laravel E-Commerce logo

Lunar:Laravel 电子商务

Laravel 的电子商务。一个开源包,将现代无头电子商务功能的力量带到 Laravel。

Lunar:Laravel 电子商务
LaraJobs logo

LaraJobs

官方 Laravel 工作板

LaraJobs
SaaSykit: Laravel SaaS Starter Kit logo

SaaSykit:Laravel SaaS 启动工具包

SaaSykit 是一款 Laravel SaaS 启动工具包,包含运行现代 SaaS 所需的所有功能。支付、漂亮的结账、管理面板、用户仪表板、身份验证、现成组件、统计数据、博客、文档等等。

SaaSykit:Laravel SaaS 启动工具包
Rector logo

Rector

您无缝升级 Laravel、降低成本和加速创新的合作伙伴,助力企业成功

Rector
MongoDB logo

MongoDB

通过将 MongoDB 和 Laravel 强力整合,增强您的 PHP 应用程序,使开发人员能够轻松高效地构建应用程序。在使用熟悉的 Eloquent API 的同时,支持事务、搜索、分析和移动用例。探索灵活的现代数据库 MongoDB 如何改变您的 Laravel 应用程序。

MongoDB
Maska is a Simple Zero-dependency Input Mask Library image

Maska 是一个简单的零依赖输入掩码库

阅读文章
Add Swagger UI to Your Laravel Application image

在您的 Laravel 应用程序中添加 Swagger UI

阅读文章
Assert the Exact JSON Structure of a Response in Laravel 11.19 image

在 Laravel 11.19 中断言响应的精确 JSON 结构

阅读文章
Build SSH Apps with PHP and Laravel Prompts image

使用 PHP 和 Laravel Prompts 构建 SSH 应用程序

阅读文章
Building fast, fuzzy site search with Laravel and Typesense image

使用 Laravel 和 Typesense 构建快速模糊网站搜索

阅读文章
Add Comments to your Laravel Application with the Commenter Package image

使用 Commenter 包为您的 Laravel 应用程序添加评论

阅读文章