简介
OPCODE是PHP编译后的二进制代码,生成的Opcode作为一种中间语言,可以帮助实现PHP源程序代码的不开源,当然,这种代码也很容易被反编译,不过对于一些简单的场景也是很足够了。
编译的基本思路是首先在php.ini中配置加载opcache扩展,并配置相关参数,然后执行一个PHP脚本遍历源代码目录,并进行编译,核心的函数是opcache_compile_file(),该函数会根据php.ini中的参数,编译并输出二进制代码。
准备工作
首先,配置PHP.INI文件中的opcache相关参数,以开启OPCACHE功能:
zend_extension=php_opcache.dll
[opcache]
; Determines if Zend OPCache is enabled
opcache.enable=1
; Determines if Zend OPCache is enabled for the CLI version of PHP
opcache.enable_cli=1
; The amount of memory for interned strings in Mbytes.
opcache.interned_strings_buffer=8
; The maximum number of keys (scripts) in the OPcache hash table.
; Only numbers between 200 and 100000 are allowed.
opcache.max_accelerated_files=8000
; When disabled, you must reset the OPcache manually or restart the
; webserver for changes to the filesystem to take effect.
opcache.validate_timestamps=0
; If disabled, all PHPDoc comments are dropped from the code to reduce the
; size of the optimized code.
opcache.save_comments=0
; Enables and sets the second level cache directory.
; It should improve performance when SHM memory is full, at server restart or
; SHM reset. The default "" disables file based caching.
opcache.file_cache=C:\MyApp\www\cache
; Enables or disables opcode caching in shared memory.
opcache.file_cache_only=0
编译
从网上得到一段编译的代码,自己根据实际情况修改了一下,代码如下:
<?php
/**
* Created by PhpStorm.
* User: Lancelot
* Date: 2018-02-09
* Time: 14:04
*/
$dir = $argv[1];
opcache_compile_files($dir);
function opcache_compile_files($dir) {
$cacheMd5 = file_get_contents("cacheMd5.txt");
foreach(new RecursiveIteratorIterator(new RecursiveDirectoryIterator($dir)) as $v) {
if(!$v->isDir() && preg_match('%\.php$%', $v->getRealPath())) {
$phpFile = $v->getRealPath();
if (filesize($phpFile) > 2) {
if (opcache_compile_file($phpFile)) {
$search = $dir;
$append = str_replace(':', '', $dir);
$repl = 'C:\BeyondScreen\www\cache\\' . $cacheMd5 . '\\' . $append;
$cachePath = str_replace($search, $repl, $phpFile) . '.bin';
if (file_exists($cachePath)) {
echo "{$phpFile}\n";
file_put_contents($phpFile, ''); //清空原来的PHP脚本
}
}
}
}
}
}
上面代码的思路是遍历传入的源文件目录,对找到的PHP文件进行编译,然后检查输出路径中是否有已经编译成功的文件,如果有,则把源PHP文件内容清空,这样服务器就会直接调用编译后的代码,而不是重新编译了。
使用方法(假设上面代码的php文件名为opcache_compile_file.php),第一个参数的值为待编译的PHP源代码目录:
php opcache_compile_file.php "C:\BeyondScreen\www\byserver"
几个问题
在实现上面的PHP代码编译过程中,遇到了一些问题,如下:
1、 源目录中的文件没有全部编译;
我们的框架是Yii2,编译后发现框架的很多代码没有编译,没有时间去查找原因,只是在编译脚本中增加了一些逻辑,只对内容不为空的PHP文件进行编译,然后编译时执行两遍编译命令来确保全部PHP文件都被编译。
2、 部分PHP文件编译失败;
大多数问题是XXX类已经存在,所以编译失败,处理方案是编译脚本中检查编译结果,对于编译失败的PHP文件不清空。