win2003下PHP使用preg_match_all導(dǎo)致apache崩潰問題的解決方法
小編的平臺(tái)是windows server 2003(32位系統(tǒng)) + Apache/2.2.9 (Win32) + PHP/5.2.17,在使用正則表達(dá)式 preg_match_all (如 preg_match_all("/ni(.*?)wo/", $html, $matches);)進(jìn)行分析匹配比較長的字符串 $html 時(shí)(大于10萬字節(jié),一般用于分析采集回來的網(wǎng)頁源碼),Apache服務(wù)器會(huì)崩潰自動(dòng)重啟。
在Apache錯(cuò)誤日志里有這樣的提示:
[Thu Apr 11 18:31:31 2013] [notice] Apache/2.2.9 (Win32) PHP/5.2.17 configured -- resuming normal operations
[Thu Apr 11 18:31:31 2013] [notice] Server built: Jun 13 2008 04:04:59
[Thu Apr 11 18:31:31 2013] [notice] Parent: Created child process 2964
[Thu Apr 11 18:31:31 2013] [notice] Disabled use of AcceptEx() WinSock2 API
[Thu Apr 11 18:31:31 2013] [notice] Child 2964: Child process is running
[Thu Apr 11 18:31:31 2013] [notice] Child 2964: Acquired the start mutex.
[Thu Apr 11 18:31:31 2013] [notice] Child 2964: Starting 350 worker threads.
[Thu Apr 11 18:31:31 2013] [notice] Child 2964: Listening on port 80.
經(jīng)過查閱Apache官方以及論壇資料后,發(fā)現(xiàn)win平臺(tái)下用正則 preg_match_all 或preg_match 分析比較長的字符串時(shí),導(dǎo)致apache崩潰重啟的原因是windows平臺(tái)下默認(rèn)分配的線程堆棧空間 ThreadStackSize 太小導(dǎo)致的。 win32默認(rèn)只有256KB,而在 linux下默認(rèn)值是 8M,這就是為什么同樣的程序在 linux平臺(tái)下正常,而在 win平臺(tái)下不正常的原因。
根據(jù)PCRE library的官方說明:256 KB 的堆棧空間對(duì)應(yīng)的pcre.recursion_limit大小應(yīng)該不超過524。
Here is a table of safe values of pcre.recursion_limit for a variety of executable stack sizes:
下面就是一張Stacksize和pcre.recursion_limit對(duì)應(yīng)的建議安全值,超過這個(gè)數(shù)值就極有可能發(fā)生堆棧溢出,apache crash:
64 MB 134217
32 MB 67108
16 MB 33554
8 MB 16777
4 MB 8388
2 MB 4194
1 MB 2097
512 KB 1048
256 KB 524
如果你沒有調(diào)整堆棧大小,就必須在使用正則的PHP頁面最開頭加入:
ini_set("pcre.recursion_limit", "524"); // PHP default is 100,000.
?>
查看具體的錯(cuò)誤可以使用下面的代碼:
if ($resultsArray === 0){
echo get_pcre_err();
}
function get_pcre_err(){
$pcre_err = preg_last_error(); // PHP 5.2 and above.
if ($pcre_err === PREG_NO_ERROR) {
$msg = 'Successful non-match.';
} else {
// preg_match error!
switch ($pcre_err) {
case PREG_INTERNAL_ERROR:
$msg = 'PREG_INTERNAL_ERROR';
break;
case PREG_BACKTRACK_LIMIT_ERROR:
$msg = 'PREG_BACKTRACK_LIMIT_ERROR';
break;
case PREG_RECURSION_LIMIT_ERROR:
$msg = 'PREG_RECURSION_LIMIT_ERROR';
break;
case PREG_BAD_UTF8_ERROR:
$msg = 'PREG_BAD_UTF8_ERROR';
break;
case PREG_BAD_UTF8_OFFSET_ERROR:
$msg = 'PREG_BAD_UTF8_OFFSET_ERROR';
break;
default:
$msg = 'Unrecognized PREG error';
break;
}
}
return($msg);
}
對(duì)于正則的修飾符 isU 說明:
s: PCRE_DOTALL,表示點(diǎn)號(hào)可以匹配換行符。
U: 表示PCRE_UNGREEDY,表示非貪婪,相當(dāng)于perl/python語言的.*?,在匹配過程中,對(duì)于.*正則,一有匹配立即執(zhí)行,而不是等.*搜索了所有字符再一一返回
在使用正則表達(dá)式時(shí),我們應(yīng)該盡量避免遞歸調(diào)用,遞歸容易導(dǎo)致堆棧溢出。比如:
/<table((?!<table).)*?<\/a>/isU 就會(huì)發(fā)生錯(cuò)誤,而使用 /<table.*?<\/a>/i 就正常。
那么如何增加win平臺(tái)下 ThreadStackSize 的大小呢? 在apache的配置文件 httpd.conf 里啟用 “Include conf/extra/httpd-mpm.conf”(刪除前面的注釋#),然后在 httpd-mpm.conf 文件里的 mpm_winnt_module 配置模塊里設(shè)置 “ThreadStackSize 8400000”即可(大約8M)。
ThreadStackSize 8400000
ThreadsPerChild 200
MaxRequestsPerChild 10000
Win32DisableAcceptEx
</IfModule>
這里需要注意的是,32位的Apache程序只能最多使用大約2GB內(nèi)存空間! 因此,ThreadStackSize 和ThreadsPerChild 的值相乘后(8M * 200)不應(yīng)該超過2G,否則無法啟動(dòng)apache,出現(xiàn)的錯(cuò)誤日志如下:
通過上面的提示,小編可以告訴大家的是在我的這臺(tái)服務(wù)器上,當(dāng)線程堆棧大小設(shè)為8M時(shí),我可以設(shè)置的線程數(shù)最多是212個(gè)。
- PHP函數(shù)preg_match_all正則表達(dá)式的基本使用詳細(xì)解析
- PHP 正則表達(dá)式之正則處理函數(shù)小結(jié)(preg_match,preg_match_all,preg_replace,preg_split)
- php中preg_match的isU代表什么意思
- PHP中preg_match函數(shù)正則匹配的字符串長度問題
- PHP中preg_match正則匹配中的/u、/i、/s含義
- PHP的preg_match匹配字符串長度問題解決方法
- PHP preg_match的匹配多國語言的技巧
- php小經(jīng)驗(yàn):解析preg_match與preg_match_all 函數(shù)
- php中使用preg_match_all匹配文章中的圖片
- php preg_match_all結(jié)合str_replace替換內(nèi)容中所有img
- PHP preg match正則表達(dá)式函數(shù)的操作實(shí)例
相關(guān)文章
新版VPS主機(jī)管理面板WDCP安裝及使用體驗(yàn)-國產(chǎn)簡單易用型VPS面板
這篇文章主要介紹了新版VPS主機(jī)管理面板WDCP安裝及使用體驗(yàn)-國產(chǎn)簡單易用型VPS面板,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07centos最小化安裝系統(tǒng)后的基本調(diào)優(yōu)及安全設(shè)置
這篇文章主要介紹了centos最小化安裝系統(tǒng)后的一些基本調(diào)優(yōu)及安全設(shè)置,需要的朋友可以參考下2013-04-04Apache中如何設(shè)置默認(rèn)首頁具體該怎么操作
這篇文章主要介紹了Apache中設(shè)置默認(rèn)首頁的方法,需要的朋友可以參考下2014-02-02PXE如何實(shí)現(xiàn)自動(dòng)化部署Linux系統(tǒng)介紹
大家好,本篇文章主要講的是PXE如何實(shí)現(xiàn)自動(dòng)化部署Linux系統(tǒng)介紹,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12詳解centos7虛擬機(jī)安裝elasticsearch5.0.x-安裝篇
本篇文章主要介紹了centos7虛擬機(jī)安裝elasticsearch5.0.x-安裝篇,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2016-12-12