OPCODE是PHP編譯後的二進制代碼,生成的Opcode做爲一種中間語言,能夠幫助實現PHP源程序代碼的不開源,固然,這種代碼也很容易被反編譯,不過對於一些簡單的場景也是很足夠了。
編譯的基本思路是首先在php.ini中配置加載opcache擴展,並配置相關參數,而後執行一個PHP腳本遍歷源代碼目錄,並進行編譯,核心的函數是opcache_compile_file(),該函數會根據php.ini中的參數,編譯並輸出二進制代碼。php
首先,配置PHP.INI文件中的opcache相關參數,以開啓OPCACHE功能:web
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文件內容清空,這樣服務器就會直接調用編譯後的代碼,而不是從新編譯了。app
使用方法(假設上面代碼的php文件名爲opcache_compile_file.php),第一個參數的值爲待編譯的PHP源代碼目錄:框架
php opcache_compile_file.php "C:\BeyondScreen\www\byserver"
在實現上面的PHP代碼編譯過程當中,遇到了一些問題,以下:
一、 源目錄中的文件沒有所有編譯;
咱們的框架是Yii2,編譯後發現框架的不少代碼沒有編譯,沒有時間去查找緣由,只是在編譯腳本中增長了一些邏輯,只對內容不爲空的PHP文件進行編譯,而後編譯時執行兩遍編譯命令來確保所有PHP文件都被編譯。
二、 部分PHP文件編譯失敗;
大多數問題是XXX類已經存在,因此編譯失敗,處理方案是編譯腳本中檢查編譯結果,對於編譯失敗的PHP文件不清空。ide