PHP細數(shù)實現(xiàn)提高并發(fā)能力的方法
本文已收錄編程學(xué)習(xí)筆記gitee。涵蓋PHP、JavaScript、Linux、Golang、MySQL、Redis和開源工具等等相關(guān)內(nèi)容。
用于生產(chǎn)環(huán)境中的PHP需要對其進行優(yōu)化,讓PHP自身發(fā)揮更好的性能,除了寫好PHP代碼,還要配置好php-fpm以及php.ini調(diào)優(yōu)。本文從內(nèi)存、OPcache、上傳、會話以及安全等方面講解php.ini的配置調(diào)優(yōu)。
PHP相對其他的編譯性語言,最大的缺點在于每次請求都會去做一些模塊解析,真正執(zhí)行的是work進程。work進程的開啟需要消耗更多的資源。同時,來一個請求都會去重新解析一些代碼,導(dǎo)致重復(fù)解析。
對于PHP的優(yōu)化,可以重點充這方面去考慮進行優(yōu)化。
內(nèi)存優(yōu)化
運行 PHP 時需要關(guān)心每個 PHP 進程要使用多少內(nèi)存,php.ini 中的memory_limit
設(shè)置用于設(shè)定單個 PHP 進程可以使用的系統(tǒng)內(nèi)存最大值。
這個設(shè)置的默認值是 128M,這對于大多數(shù)中小型 PHP 應(yīng)用來說或許合適,不過,如果運行的是微型 PHP 應(yīng)用,可以降低這個值,以便節(jié)省系統(tǒng)資源,反之,如果運行的是內(nèi)存集中型 PHP 應(yīng)用,可以增加這個值。這個值的大小由可用的系統(tǒng)內(nèi)存決定,確定給 PHP 分配多少值是一門藝術(shù),決定給 PHP 分配多少內(nèi)存,以及能負擔(dān)起多少個 PHP-FPM 進程時,可以根據(jù)以下維度信息進行判斷:
- 一共可以分配給 PHP 多少內(nèi)存?以一個 2G 內(nèi)存的 VPS 為例,這臺設(shè)備中可能還運行了其他進程,如 MySQL、Nginx 等,那么留 512M 給 PHP 是合適的。
- 每個 PHP 進程平均耗費多少內(nèi)存?這個要監(jiān)控進程的內(nèi)存使用量,可以使用命令行命令
top
,也可以在 PHP 腳本中調(diào)用memory_get_peak_usage()
函數(shù),不管使用哪種方式,都要多次運行同一個腳本,然后取內(nèi)存消耗的平均值。 - 能負擔(dān)起多少個 PHP-FPM 進程?假設(shè)我給 PHP 分配了 512M 內(nèi)存,每個 PHP 進程平均耗費 15M 內(nèi)存,那么可以負擔(dān)起 34 個 PHP-FPM 進程。
有足夠的系統(tǒng)資源嗎?最后還需要確認有足夠的系統(tǒng)資源運行 PHP 應(yīng)用并處理預(yù)期的流量。具體的PHP配置信息可以參考php-fpm.config配置文件。
; Time limit for child processes to wait for a reaction on signals from master.
; Available units: s(econds), m(inutes), h(ours), or d(ays)
; Default Unit: seconds
; Default Value: 0
;process_control_timeout = 0; The maximum number of processes FPM will fork. This has been designed to control
; the global number of processes when using dynamic PM within a lot of pools.
; Use it with caution.
; Note: A value of 0 indicates no limit
; Default Value: 0
; process.max = 128; Specify the nice(2) priority to apply to the master process (only if set)
; The value can vary from -19 (highest priority) to 20 (lowest priority)
; Note: - It will only work if the FPM master process is launched as root
; - The pool process will inherit the master process priority
; unless specified otherwise
; Default Value: no set
; process.priority = -19; Send FPM to background. Set to 'no' to keep FPM in foreground for debugging.
; Default Value: yes
daemonize = no
php-fpm有三種運行模式,分別是固定進程數(shù)、按需進程數(shù)、完全動態(tài)進程數(shù)。
- 按需進程數(shù),默認初始化幾個進程數(shù),如果進去量過大,動態(tài)創(chuàng)建一些新的進程數(shù),等請求結(jié)束之后,新創(chuàng)建的進程數(shù)在銷毀掉。
- 固定進程數(shù),默認固定幾個進程,如果進程數(shù)不夠的情況時,新的請求處于等待中,直到其他的進程處理完畢才會處理新的請求。
- 完全動態(tài)進程數(shù),表示完全由請求量控制,來一個請求創(chuàng)建一個進程,處理完畢在銷毀掉。
開啟Zend OPcache性能加速
確定要分配多少內(nèi)存后,就可以配置 PHP 的 Zend OPcache 擴展。OPcache主要是將一些代碼解析成字節(jié)碼,在后續(xù)的請求中就無需重復(fù)去解析、編譯這部分代碼。減少編譯、解析的過程,也能提高PHP的處理速度。
PHP5.5.0+內(nèi)置了這個擴展,下面羅列幾項必要的配置信息:
opcache.memory_consumption = 64
:為操作碼緩存分配的內(nèi)存(單位是MB),分配的內(nèi)存量應(yīng)該可以保存應(yīng)用中所有 PHP 腳本編譯得到的操作碼,這個值根據(jù)應(yīng)用的體量可以設(shè)置成不同大小的值。
opcache.interned_strings_buffer = 16
:用來存儲駐留字符串的內(nèi)存量(單位是MB),什么是駐留字符串呢?PHP 解釋器在背后會找到相同字符串的多個實例,把這個字符串保存在內(nèi)存中,如果再次使用相同的字符串,PHP 解釋器會使用指針,這么做的目的是節(jié)省內(nèi)存。默認情況下,PHP 駐留字符串會隔離在各個 PHP 進程中,這個設(shè)置能讓 PHP-FPM 進程池把所有進程駐留字符串存儲到共享的緩沖區(qū)中,以便在 PHP-FPM 進程池中的多個進程之間引用駐留字符串,這樣能節(jié)省更多內(nèi)存。
opcache.max_accelerated_files = 4000
:操作碼緩存中最多能存儲多少個 PHP 腳本,這個值的區(qū)間是 2000 到 100000 之間,這個值一定要比 PHP 應(yīng)用中的文件數(shù)大。
opcache.validate_timestamps = 1
:這個設(shè)置的值為1時,經(jīng)過一段時間后 PHP 會檢查 PHP 腳本的內(nèi)容是否有變化,檢查的時間間隔由opcache.revalidate_freq設(shè)置指定。如果這個設(shè)置的值為0,PHP 不會檢查 PHP 腳本的內(nèi)容是否有變化,我們必須自己動手清除緩存的操作碼。建議在開發(fā)環(huán)境中設(shè)置為1,生產(chǎn)環(huán)境中設(shè)置為0。
opcache.revalidate_freq = 0
:設(shè)置多久(單位是秒)檢查一次 PHP 腳本內(nèi)容是否有變化。設(shè)置為0秒的含義是僅當(dāng)opcache.validate_timestamps設(shè)置為1時,才會在每次請求時都重新驗證 PHP 文件,因此,在開發(fā)環(huán)境中每次都會重新驗證 PHP 文件,在生產(chǎn)環(huán)境中則不驗證。
opcache.fast_shutdown = 1
:這么設(shè)置能讓操作碼使用更快的停機步驟,把對象析構(gòu)和內(nèi)存釋放交給 Zend Engine 的內(nèi)存管理器完成。
文件上傳
如果你的應(yīng)用允許上傳文件,最好設(shè)置最大能上傳的文件大小。除此之外,最好還要設(shè)置最多能同時上傳多少個文件:
file_uploads = 1 upload_max_filesize = 10M max_file_uploads = 3
默認情況下,PHP 允許在單次請求中上傳 20 個文件,上傳的文件最大為 2MB,這里我設(shè)置為單次請求最多只能上傳 3 個文件,每個文件最大為 10MB,這個值不要設(shè)置太大,否則會出現(xiàn)超時。
注:如果非要上傳大文件,Web 服務(wù)器的配置也要做相應(yīng)調(diào)整。除了在 php.ini 中設(shè)置之外,還要調(diào)整 Nginx 虛擬主機配置中的 client_max_body_size
設(shè)置。
此外,如果是上傳特大文件,我建議使用Webuploader專門的上傳組件,前端對大文件進行切片,后端php對分片數(shù)據(jù)進行合并還原文件。有關(guān)WebUploader應(yīng)用請參考本站文章:功能強大的文件上傳組件-WebUploader。
執(zhí)行時間
max_execution_time 用于設(shè)置單個 PHP 進程在終止之前最長可運行時間。這個設(shè)置默認是 30 秒,建議將其設(shè)置為 5 秒:
max_execution_time = 5
在 PHP 腳本中可以調(diào)用set_limit_time()
函數(shù)覆蓋這個設(shè)置。
假設(shè)我們想要生成報告,并把結(jié)果制作成 PDF 文件,這個任務(wù)可能要花 10 分鐘才能完成,而我們肯定不想讓 PHP 請求等待 10 分鐘,我們應(yīng)該單獨編寫一個 PHP 文件,讓其在單獨的后臺進程中執(zhí)行,Web 應(yīng)用只需幾毫秒就可以派生一個單獨的后臺進程,然后返回 HTTP 響應(yīng)。
實際上,我們在跑需要消耗大量時間來完成的任務(wù),一般采用后臺進程方式,比如我們可以使用PHP的swoole擴展來生成報表、批量發(fā)送郵件耗時長的任務(wù)。
處理會話
PHP默認的情況是將會話產(chǎn)生的信息存在磁盤中,例如所謂的session信息。在創(chuàng)建和讀取session時,都會對磁盤進行I/O操作,讀寫磁盤其實是比較耗時的一個操作。并且session不方便做分布式應(yīng)用的會話機制處理。推薦可以放在Redis、memcached這樣的內(nèi)存性服務(wù)中,讀寫速度快,并且可以做分布式會話機制處理。
下面舉例將session這樣的信息,存儲在memcached內(nèi)存中。
session.save_handler = "memcached" session.save_path = "服務(wù)地址:端口號"
緩沖區(qū)
如果是在較少的塊中發(fā)送更多數(shù)據(jù),而不是在較多的塊中發(fā)送較少的數(shù)據(jù),那么網(wǎng)絡(luò)的效率會更高,也就是說,在較少的片段中把內(nèi)容傳遞給訪問者的瀏覽器,能減少 HTTP 請求總數(shù)。
因此,我們要讓 PHP 緩沖輸出,默認情況下,PHP 已經(jīng)啟用了輸出緩沖功能,PHP 緩沖 4096 字節(jié)的輸出之后才會把內(nèi)容發(fā)送給 Web 服務(wù)器,推薦配置如下:
output_buffering = 4096 implicit_flush = false
如果想要修改輸出緩沖區(qū)的大小,確保使用的值是4(32位系統(tǒng))或8(64位系統(tǒng))的倍數(shù)。
安全設(shè)置
open_basedir
:使用open_basedir選項能夠控制PHP腳本只能訪問指定的目錄,這樣能夠避免PHP腳本訪問不應(yīng)該訪問的文件,一定程度上限制了phpshell的危害。我們一般可以設(shè)置為只能訪問網(wǎng)站目錄:
open_basedir = /data/www
disable_functions
:一般我們要禁止系統(tǒng)函數(shù)和禁止任何文件和目錄的操作,如:
disable_functions = '.....'
expose_php = Off
:將此項設(shè)置為false即不會再header頭輸出PHP版本信息。
display_errors = Off
:生產(chǎn)環(huán)境中,我們應(yīng)該禁止錯誤提示,如果是本地開發(fā)環(huán)境,可以設(shè)置為On。
log_errors = On
:建議在關(guān)閉display_errors后能夠把錯誤信息記錄下來,便于查找服務(wù)器運行的原因。
error_log
:設(shè)置PHP錯誤日志存放的目錄。
到此這篇關(guān)于PHP細數(shù)實現(xiàn)提高并發(fā)能力的方法的文章就介紹到這了,更多相關(guān)PHP并發(fā)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
PHP字符串函數(shù)系列之nl2br(),在字符串中的每個新行 (\n) 之前插入 HTML 換行符br
nl2br() 函數(shù)在字符串中的每個新行 (\n) 之前插入 HTML 換行符 (br)。2011-11-11php實現(xiàn)使用正則將文本中的網(wǎng)址轉(zhuǎn)換成鏈接標(biāo)簽
本文給大家分享一段php中使用正則表達式將網(wǎng)址轉(zhuǎn)換成A鏈接的函數(shù)代碼,十分簡潔實用,這里推薦給大家2014-12-12解析php函數(shù)method_exists()與is_callable()的區(qū)別
本篇文章是對php中method_exists()與is_callable()的區(qū)別進行了詳細的分析介紹,需要的朋友參考下2013-06-06