本篇文章帶大家了解一下OPCache,詳細(xì)介紹一下使用OPCache提升PHP的性能的方法。有一定的參考價值,有需要的朋友可以參考一下,希望對大家有所幫助。
對于 PHP 這樣的解釋型語言來說,每次的運行都會將所有的代碼進(jìn)行一次加載解析,這樣一方面的好處是代碼隨時都可以進(jìn)行熱更新修改,因為我們不需要編譯。但是這也會帶來一個問題,那就是無法承載過大的訪問量。畢竟每次加載解析再釋放,都會增加 CPU 的負(fù)擔(dān),通常一臺 8核16G 的服務(wù)器在2、3000并發(fā)左右 CPU 就能達(dá)到60%以上的使用率。而且如果你使用的是類似于 Laravel 這種大型的框架,效率將更加低下。這個時候,我們通常會通過增加服務(wù)器數(shù)量來做負(fù)載均衡,從而達(dá)到減輕服務(wù)器壓力的效果。不過,這樣做的成本又會增加許多。那么,有沒有什么優(yōu)化的方案呢?
鳥哥在他的博客中針對 PHP7 的優(yōu)化的一篇文章中,第一條建議就是開啟 OPcache 。當(dāng)然,另外一個方案就是使用 Swoole 。關(guān)于 Swoole 的內(nèi)容我們將來再說,今天,我們先學(xué)習(xí)學(xué)習(xí) OPcache 。
什么是 OPcache
OPcache 通過將 PHP 腳本預(yù)編譯的字節(jié)碼存儲到共享內(nèi)存中來提升 PHP 的性能, 存儲預(yù)編譯字節(jié)碼的好處就是 省去了每次加載和解析 PHP 腳本的開銷。
這是 PHP 文檔中關(guān)于 OPcache 的簡介,也就是說,OPcache 節(jié)約了每次加載和解析的步驟,將第一次解析編譯后的腳本字節(jié)碼緩存到系統(tǒng)的共享內(nèi)存中。其實,這就類似于一個不完全的編譯。
類似于 Java 之類的語言,都是要打包編譯之后才能上線運行的,比如打包成一個 jar包 。C++ 或 C# 可以打包成一個 .dll 或 .exe 。這些打包之后的文件就是編譯完成的文件,將它們運行起來后一般會一直保持運行狀態(tài),也就是會成為一個常駐進(jìn)程,它們的代碼就進(jìn)入內(nèi)存中了。在程序運行的時候,不需要再進(jìn)行解釋或編譯,自然速度就要快很多。而 OPcache 也是起到類似的作用。只不過它并不是完全的一套編譯流程,我們還是依賴的 PHP-FPM 來運行腳本,只不過在開啟 OPcache 后,PHP-FPM 會先從內(nèi)存中查找是否已經(jīng)有相關(guān)的已經(jīng)緩存的字節(jié)碼在內(nèi)存中了,如果有的話就直接取用,如果沒有的話,會再次進(jìn)行解釋編譯后緩存下來。另外,OPcache 是針對文件的,也就是說,一個文件如果是新增加進(jìn)來的,只有運行過它才會緩存,如果沒有運行過,它并不在當(dāng)前的共享內(nèi)存中。
推薦學(xué)習(xí):《PHP視頻教程》
安裝 Opcache
OPcache 已經(jīng)是 PHP 的官方擴展并隨安裝包一起發(fā)布了,所以,我們可以在編譯安裝 PHP 時使用 –enable-opcache 來開啟擴展,它已經(jīng)是默認(rèn)擴展。也可以在未安裝 OPcache 的系統(tǒng)中使用安裝包中的文件來進(jìn)行安裝。
cd php-7.4.4/ext/opcache/ phpize ./configure make && make install
需要注意的是, OPcache 和 Xdebug 在生產(chǎn)環(huán)境中盡量不要一起使用。本身 Xdebug 就是不推薦在生產(chǎn)環(huán)境中使用的,如果一定需要同時使用的話,需要先加載 OPcache ,然后再加載 Xdebug 。
擴展安裝后,在 php.ini 文件中打開擴展。需要注意的是,OPcache 擴展是 Zend 擴展包,所以我們需要打開的是 Zend 擴展。
zend_extension=opcache.so
另外,還需要啟用它。
opcache.enable=1
當(dāng)開啟了 OPcache 之后,我們再更新代碼將會發(fā)現(xiàn)剛剛更新的代碼不是我們最新的代碼。這是因為代碼已經(jīng)被緩存了,就像 Java 一樣,我們需要重啟服務(wù)才行。那么 PHP 這邊重啟的是什么呢?當(dāng)然就是重啟下我們的 PHP-FPM 就可以了,直接使用 kill -USR2 命令去重啟主進(jìn)程就行了。這里也給出一個快速重啟的命令。
ps -ef | grep "php-fpm: master" | grep -v grep | cut -c 9-15 | xargs kill -USR2
感謝知乎大佬的指正,重啟 PHP-FPM 不是最佳方案,應(yīng)該使用 opcache_reset() 手動重啟,或者通過 php.ini 文件的配置 opcache.validate_timestamps + opcache.revalidate_freq 自動間隔編譯,或者通過 opcache_compile_file() 來直接重新編譯修改過的文件
ab 測試效果
我們進(jìn)行測試的內(nèi)容是測試環(huán)境的一臺2核4G的服務(wù)器,使用的 PHP 版本是 PHP7.4 ,正常的 Nginx 及 PHP 配置, ulimit 也都開到了最大。代碼只是簡單的輸出了一行文字,不過我們使用的是一個簡單的 mvc 框架 ,也就是說這段代碼運行起來至少也會加載幾個文件,而不是簡簡單單的一個文件。
首先我們來看未開啟 OPcache 的情況。
接下來是開啟了 OPcache 的情況。
很明顯,性能有了很大的提高。不僅速度快了很多,吞吐率也是直接上升了幾倍。當(dāng)然,這只是非常簡單的一個測試,不過總體看來,確實對單機的性能提升有很大的幫助。最最主要的是,同樣的并發(fā)情況下,CPU 資源也比未開啟的狀態(tài)下低了70%。
配置參考
在 PHP 的官方文檔中,已經(jīng)為我們給出了一套默認(rèn)的 OPcache 在 php.ini 中的配置。經(jīng)過測試,基本沒什么問題,當(dāng)然,現(xiàn)在還沒有在生產(chǎn)環(huán)境中使用過,還需要進(jìn)行