很多文章說mod_php不能和apache mpm worker或者event共用,這也許是在apache 2.2版本時代流傳下來的說法。自去年編譯apache 2.4以來,我就一直在使用mpm event,而且根據我編譯php 5.5的參數來看,我就是把php做為apache的一個內置模塊。這樣奇葩的配置存在一年多以后,我終於決定從mod_php過渡到php-fpm。
自apache 2.4.9以來,php-fpm協同apache一起工作所需要的配置沒有以前那么復雜了,這是讓我決定過渡的主要原因之一。
重新編譯php,和之前編譯參數差不太多,只需要去掉“--with-apxs2=/usr/local/apache2/bin/apxs”,加入“--enable-fpm”:
./configure --enable-shared --with-libxml-dir --with-gd --with-openssl --enable-mbstring --with-mcrypt --with-mysqli --with-mysql --enable-opcache --enable-mysqlnd --enable-zip --with-zlib-dir --with-pdo-mysql --with-jpeg-dir --with-freetype-dir --with-curl --without-pdo-sqlite --without-sqlite3 --enable-fpm
隨后make、make install即可。為了讓centos可以開機自動啟動php-fpm,我們需要:
cp sapi/fpm/init.d.php-fpm /etc/init.d/php-fpm chmod o+x /etc/init.d/php-fpm chkconfig --add php-fpm
讓php-fpm的配置文件生效:
mv /usr/local/etc/php-fpm.conf.default /usr/local/etc/php-fpm.conf
根據自己的需要修改這個conf文件,我個人喜歡:
user = daemon group = daemon listen = /dev/shm/php-fpm.sock listen.owner = daemon listen.group = daemon pm = static pm.max_children = 4 ;見下面解釋 pm.max_requests = 2048
如果是多核cpu的vps或者服務器,上面的數值等於cpu數量即可;如果是單核的vps,那么pm.max_children = 2,即可達到一定的優化效果。
再說說apache這邊需要的一點設置。由於我是從mod_php到php-fpm的,所以之前httpd.conf中的這幾行需要注解掉:
LoadModule php5_module modules/libphp5.so <FilesMatch \.php$> SetHandler application/x-httpd-php </FilesMatch>
然后去掉mod_proxy.so和mod_proxy_fcgi.so之前的注解,確保他們被apache加載。
如果php-fpm使用的是TCP socket,那么在httpd.conf末尾加上:
<FilesMatch \.php$> SetHandler "proxy:fcgi://127.0.0.1:9000" </FilesMatch>
如果用的是unix socket,那么httpd.conf末尾加上:
<Proxy "unix:/dev/shm/php-fpm.sock|fcgi://php-fpm"> ProxySet disablereuse=off </Proxy> <FilesMatch \.php$> SetHandler proxy:fcgi://php-fpm </FilesMatch>
至此在apapche上配置php-fpm結束:
service php-fpm start
service httpd restart
人品好的人,是不會看見出錯信息的。如果訪問服務器上的php文件出現"file not found."十之八九是php-fpm.conf中的user和group沒有讀寫權限,修改成apache所使用的user和group即可。
最后附上在虛擬機上ab測試apache mod_php和php-fpm的結果,測試頁面是我這個博客的主頁。測試過程中cpu 100%是性能瓶頸,內存有余:
- mod_php:Requests per second: 54.38 [#/sec]
- php-fpm:Requests per second: 67.55 [#/sec]
php-fpm在性能上的提升還是很明顯的,而且內存占用明顯小於mod_php。
----
php-fpm的啟動參數
#測試php-fpm配置 /usr/local/php/sbin/php-fpm -t /usr/local/php/sbin/php-fpm -c /usr/local/php/etc/php.ini -y /usr/local/php/etc/php-fpm.conf -t #啟動php-fpm /usr/local/php/sbin/php-fpm /usr/local/php/sbin/php-fpm -c /usr/local/php/etc/php.ini -y /usr/local/php/etc/php-fpm.conf #關閉php-fpm kill -INT `cat /usr/local/php/var/run/php-fpm.pid` #重啟php-fpm kill -USR2 `cat /usr/local/php/var/run/php-fpm.pid`
php-fpm.conf重要參數詳解
pid = run/php-fpm.pid #pid設置,默認在安裝目錄中的var/run/php-fpm.pid,建議開啟 error_log = log/php-fpm.log #錯誤日志,默認在安裝目錄中的var/log/php-fpm.log log_level = notice #錯誤級別. 可用級別為: alert(必須立即處理), error(錯誤情況), warning(警告情況), notice(一般重要信息), debug(調試信息). 默認: notice. emergency_restart_threshold = 60 emergency_restart_interval = 60s #表示在emergency_restart_interval所設值內出現SIGSEGV或者SIGBUS錯誤的php-cgi進程數如果超過 emergency_restart_threshold個,php-fpm就會優雅重啟。這兩個選項一般保持默認值。 process_control_timeout = 0 #設置子進程接受主進程復用信號的超時時間. 可用單位: s(秒), m(分), h(小時), 或者 d(天) 默認單位: s(秒). 默認值: 0. daemonize = yes #后台執行fpm,默認值為yes,如果為了調試可以改為no。在FPM中,可以使用不同的設置來運行多個進程池。 這些設置可以針對每個進程池單獨設置。 listen = 127.0.0.1:9000 #fpm監聽端口,即nginx中php處理的地址,一般默認值即可。可用格式為: 'ip:port', 'port', '/path/to/unix/socket'. 每個進程池都需要設置. listen.backlog = -1 #backlog數,-1表示無限制,由操作系統決定,此行注釋掉就行。backlog含義參考:http://www.3gyou.cc/?p=41 listen.allowed_clients = 127.0.0.1 #允許訪問FastCGI進程的IP,設置any為不限制IP,如果要設置其他主機的nginx也能訪問這台FPM進程,listen處要設置成本地可被訪問的IP。默認值是any。每個地址是用逗號分隔. 如果沒有設置或者為空,則允許任何服務器請求連接 listen.owner = www listen.group = www listen.mode = 0666 #unix socket設置選項,如果使用tcp方式訪問,這里注釋即可。 user = www group = www #啟動進程的帳戶和組 pm = dynamic #對於專用服務器,pm可以設置為static。 #如何控制子進程,選項有static和dynamic。如果選擇static,則由pm.max_children指定固定的子進程數。如果選擇dynamic,則由下開參數決定: pm.max_children #,子進程最大數 pm.start_servers #,啟動時的進程數 pm.min_spare_servers #,保證空閑進程數最小值,如果空閑進程小於此值,則創建新的子進程 pm.max_spare_servers #,保證空閑進程數最大值,如果空閑進程大於此值,此進行清理 pm.max_requests = 1000 #設置每個子進程重生之前服務的請求數. 對於可能存在內存泄漏的第三方模塊來說是非常有用的. 如果設置為 '0' 則一直接受請求. 等同於 PHP_FCGI_MAX_REQUESTS 環境變量. 默認值: 0. pm.status_path = /status #FPM狀態頁面的網址. 如果沒有設置, 則無法訪問狀態頁面. 默認值: none. munin監控會使用到 ping.path = /ping #FPM監控頁面的ping網址. 如果沒有設置, 則無法訪問ping頁面. 該頁面用於外部檢測FPM是否存活並且可以響應請求. 請注意必須以斜線開頭 (/)。 ping.response = pong #用於定義ping請求的返回相應. 返回為 HTTP 200 的 text/plain 格式文本. 默認值: pong. request_terminate_timeout = 0 #設置單個請求的超時中止時間. 該選項可能會對php.ini設置中的'max_execution_time'因為某些特殊原因沒有中止運行的腳本有用. 設置為 '0' 表示 'Off'.當經常出現502錯誤時可以嘗試更改此選項。 request_slowlog_timeout = 10s #當一個請求該設置的超時時間后,就會將對應的PHP調用堆棧信息完整寫入到慢日志中. 設置為 '0' 表示 'Off' slowlog = log/$pool.log.slow #慢請求的記錄日志,配合request_slowlog_timeout使用 rlimit_files = 1024 #設置文件打開描述符的rlimit限制. 默認值: 系統定義值默認可打開句柄是1024,可使用 ulimit -n查看,ulimit -n 2048修改。 rlimit_core = 0 #設置核心rlimit最大限制值. 可用值: 'unlimited' 、0或者正整數. 默認值: 系統定義值. chroot = #啟動時的Chroot目錄. 所定義的目錄需要是絕對路徑. 如果沒有設置, 則chroot不被使用. chdir = #設置啟動目錄,啟動時會自動Chdir到該目錄. 所定義的目錄需要是絕對路徑. 默認值: 當前目錄,或者/目錄(chroot時) catch_workers_output = yes #重定向運行過程中的stdout和stderr到主要的錯誤日志文件中. 如果沒有設置, stdout 和 stderr 將會根據FastCGI的規則被重定向到 /dev/null . 默認值: 空.