尧图网站建设 尧图网络
  • 首页
  • 关于我们
  • 服务项目
  • 案例展示
  • 建站流程
  • 资讯中心
  • 联系我们
首页/资讯中心/详情

PHP 如何利用 Opcache 来实现保护源码

PHP 如何利用 Opcache 来实现保护源码
📅 发布时间:2026/6/23 1:14:34

PHP 如何利用 Opcache 来实现保护源码

感兴趣得可以试试看!!!

要求

  • 不用 IonCube(或类似的)。不知道这是啥的话,就是加密 PHP 代码但还能运行的工具。问题是太贵了。😅
  • 性能要好,PHP 原生支持。

原文链接 PHP 如何利用 Opcache 来实现保护源码

后来想到,PHP 有个"opcache"功能,能把源码编译成操作码(机器语言)在 Zend VM 上跑,跟 Java 差不多 😃 厉害的是,这样既保护了代码,又提升了性能!

开始干活。要让这套方案跑起来,得把代码打包成镜像(就是个只读的存储,跟系统其他部分隔离开),因为 opcache 是全局生效的,不管哪个 PHP 项目。最好的工具就是 Docker。(Docker 比虚拟机轻量多了,分发部署都很方便)。

这次用 Laravel 做例子。为啥选它?因为组件多,依赖库也多,能遇到各种坑,学到的东西也多。
Laravel 默认目录结构

一般来说,我们的核心代码都在 /app 目录里,这部分需要保护。其他目录像 /vendor 都是开源库,不用管。

具体步骤

第一步,在项目根目录建个 warm-opcache.php 文件。这玩意儿会调用 opcache_compile_file() 手动让 PHP 编译代码。

<?php
$directory = new RecursiveDirectoryIterator('/var/www/app'); # 我们用 /var/www
$iterator = new RecursiveIteratorIterator($directory);foreach ($iterator as $file) {if (pathinfo($file, PATHINFO_EXTENSION) === 'php') {echo "编译中: {$file}\n";opcache_compile_file($file);}
}

第二步,建个 empty-preserve-time.sh 脚本(记得 chmod +x 给执行权限)。这个脚本会把 PHP 文件内容清空,但保留时间戳。为啥要保留时间戳?因为文件修改时间一变,opcache 就会重新加载。

#!/bin/bashfor file in $(find ./app -type f -name "*.php"); dotimestamp=$(stat -c %Y "$file")  # 获取修改时间(从纪元开始的秒数): > "$file"                      # 清空文件touch -d "@$timestamp" "$file"   # 恢复原始时间戳
done

第三步,把 zz-opcache.ini 配置文件放到 /usr/local/etc/php/conf.d 目录(或者你系统的 conf.d 在哪就放哪)。(记得先装好 PHP 的 opcache 扩展)

opcache.enable=1
opcache.enable_cli=1
opcache.validate_timestamps=1
opcache.revalidate_freq=10
opcache.file_cache=/var/www/.opcache
opcache.file_cache_only=1

⚠️ 重要:先把代码 commit 或者备份!下面的操作会删除文件内容!

接下来就是见证奇迹的时刻了。先跑 php warm-opcache.php,再跑 empty-preserve-time.sh,文件内容会被清空,但 /app 目录结构还在,Laravel 项目照样能跑。不信你试试!

这套方法对任何 PHP 项目都管用,不管你用 PSR-4 还是简单的 require()。Laravel 用的是 PSR-4。

不错,概念验证成功。下一步就是打包,要能分发到客户的服务器上。(就像 Go 能编译成 .exe 一样)

直接上 Dockerfile。(这个 Dockerfile 没做层优化,主要是为了好理解)

FROM php:8.3-fpm-alpine # 根据需要修改# 添加更多 pecl install 或 docker-php-ext-install
# 来安装项目需要的扩展# 启用 opcache
RUN docker-php-ext-install opcacheWORKDIR /var/www
RUN mkdir -p /var/www/.opcache# 复制源码
COPY app ./app
COPY artisan ./artisan
COPY bootstrap ./bootstrap
COPY database ./database
COPY config ./config
COPY public ./public
COPY resources ./resources
COPY routes ./routes
COPY storage ./storage
COPY composer.* .# 安装 ini 文件
COPY zz-opcache.ini /usr/local/etc/php/conf.d# Laravel 的 composer install 需要 .env
# 我们复制一个假的 .env
COPY .env.example .env# 安装 PHP 依赖(不要把这行移到上面)
RUN composer install --no-dev --optimize-autoloader# 编译并删除 /app 中的源码
RUN php warm-opcache.php
RUN ./empty-preserve-time.sh# 恭喜!你的代码已经被清除了!
# 如果不信,你可以 `ls` 你的 /app 目录并 `cat` 它# 如果需要,你可以创建一个 ENTRYPOINT 脚本,也可以执行
# ./artisan queue:work, 或 ./artisan schedule:work
CMD ["./artisan serve"]

现在,你可以 docker build 并 docker push 到你的注册服务器,然后从客户的本地服务器 pull,而不用裸露地交付代码!当你有更新时,简单的 docker pull 就能节省很多时间!

可能有人会问,我们能删除 /app 目录而不是留空吗?不行。因为"opcache"会检查文件是否存在。

额外收获!

上面的 Dockerfile 不安全。为什么?因为 Docker 在每个阶段都使用层,意味着当你 COPY app ./app 时,它实际上复制了你未保护的代码,并创建了一个层。Docker 专家可以轻松解开这些层,获取你的原始代码。

解决方案是使用多阶段构建。这是修订后的 Dockerfile。注意我们在第 1 行添加了 as build。

FROM php:8.3-fpm-alpine as build # 根据需要修改# 添加更多 pecl install 或 docker-php-ext-install
# 来安装项目需要的扩展# 启用 opcache
RUN docker-php-ext-install opcacheWORKDIR /var/www
RUN mkdir -p /var/www/.opcache# 复制源码
COPY app ./app
COPY artisan ./artisan
COPY bootstrap ./bootstrap
COPY database ./database
COPY config ./config
COPY public ./public
COPY resources ./resources
COPY routes ./routes
COPY storage ./storage
COPY composer.* .# 安装 ini 文件
COPY zz-opcache.ini /usr/local/etc/php/conf.d# Laravel 的 composer install 需要 .env
# 我们复制一个假的 .env
COPY .env.example .env# 安装 PHP 依赖(不要把这行移到上面)
RUN composer install --no-dev --optimize-autoloader# 编译并删除 /app 中的源码
RUN php warm-opcache.php
RUN ./empty-preserve-time.sh# 恭喜!你的代码已经被清除了!
# 如果不信,你可以 `ls` 你的 /app 目录并 `cat` 它# ======== 这里是多阶段层构建 ===========
FROM php:8.3-fpm-alpine # 根据需要修改WORKDIR /var/www# (重复上面完全相同的步骤)
# 添加更多 pecl install 或 docker-php-ext-install
# 来安装项目需要的扩展# 启用 opcache
RUN docker-php-ext-install opcache# 安装 ini 文件
COPY zz-opcache.ini /usr/local/etc/php/conf.d# 从 `build` 复制清空的文件和 opcache 代码到这里
COPY --from=build /var/www .# 如果需要,你可以创建一个 ENTRYPOINT 脚本,也可以执行
# ./artisan queue:work, 或 ./artisan schedule:work
CMD ["./artisan serve"]

相关新闻

  • 【操作系统】从实模式到保护模式,
  • Flutter CSV导入导出:大数据处理与用户体验优化
  • NET 中 Async/Await 的演进:从状态机到运行时优化的 Continuation

最新新闻

  • Vue 3国际化实战:vue-i18n核心原理与工程化落地
  • Java FutureTask 深度解析:状态机、超时控制与线程中断原理
  • Qwen3.5+llama.cpp实测:216G显存跑262K上下文与120 tokens/s推理
  • RTA广告技术解析:从实时API原理到电商金融实战部署
  • Prisma + PostgreSQL 生产级落地指南:从连接配置到向量搜索
  • FEC以太网控制器:缓冲区描述符机制与嵌入式网络驱动开发实战

日新闻

  • Arduino-ESP32项目深度解析:解锁隐藏芯片支持与架构演进
  • 2026年 系统窗厂家/品牌推荐榜单:隔音系统窗+高端系统门窗的核心优势与选购指南 - 品牌发掘
  • NVBench:首个双语非言语发声语音合成评测基准详解与实践

周新闻

  • Visual C++运行库修复终极指南:5分钟快速解决Windows软件启动错误
  • 手把手教你构建统计局地区经济数据爬虫:从环境搭建到数据持久化全指南
  • 2026多Agent深度解析:用AI团队替代单一模型,四种架构实战落地

月新闻

  • 【总结】入门篇:50句话让你记住架构核心概念
  • WeChatMsg技术方案解析:实现Mac微信数据自主管理的完整解决方案
  • WeChatMsg:革新性微信数据备份方案,打造你的专属数字记忆库

关于尧图

  • 公司简介
  • 团队介绍
  • 企业文化
  • 荣誉资质

服务项目

  • 定制开发
  • 电商建站
  • UI 设计
  • 运维服务

快速链接

  • 案例展示
  • 建站流程
  • 常见问题
  • 资讯中心

联系方式

  • 📍北京市朝阳区互联网产业园 A 座 10 层
  • 📞400-888-8888
  • ✉️contact@rkmt.cn
  • 🕐周一至周日 9:00-21:00

© 2024 北京尧图网络科技有限公司 版权所有 | 京 ICP 备 XXXXXXXX 号