本文章來給大家介紹在win2003使用preg_match_all導致apache崩潰解決辦法,有碰到此類問題的朋友可進入參考參考。
Apache/2.2.9 (Win32) + PHP/5.2.17,在使用正則表達式 preg_match_all (如 preg_match_all("/ni(.*?)wo/", $html, $matches);)進行分析匹配比較長的字符串 $html 時(大於10萬字節,一般用於分析采集回來的網頁源碼),Apache服務器會崩潰自動重啟。
在Apache錯誤日志裡有這樣的提示:
[Thu Apr 11 18:31:31 2013] [notice] Parent: child process exited with status 128 -- Restarting.
[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.
經過查閱Apache官方以及論壇資料後,發現win平台下用正則 preg_match_all 或preg_match 分析比較長的字符串時,導致apache崩潰重啟的原因是windows平台下默認分配的線程堆棧空間 ThreadStackSize 太小導致的。 win32默認只有256KB,而在 linux下默認值是 8M,這就是為什麼同樣的程序在 linux平台下正常,而在 win平台下不正常的原因。
根據PCRE library的官方說明:256 KB 的堆棧空間對應的pcre.recursion_limit大小應該不超過524。
Here is a table of safe values of pcre.recursion_limit for a variety of executable stack sizes:
下面就是一張Stacksize和pcre.recursion_limit對應的建議安全值,超過這個數值就極有可能發生堆棧溢出,apache crash:
Stacksize pcre.recursion_limit
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
如果你沒有調整堆棧大小,就必須在使用正則的PHP頁面最開頭加入:
<?php
ini_set("pcre.recursion_limit", "524"); // PHP default is 100,000.
?>
查看具體的錯誤可以使用下面的代碼:
代碼如下 復制代碼 $resultsArray = preg_match_all("/table.*?<a>/isU", $html, $contents); 對於正則的修飾符 isU 說明:
i: 表示in-casesensitive,即大小寫不敏感
s: PCRE_DOTALL,表示點號可以匹配換行符。
U: 表示PCRE_UNGREEDY,表示非貪婪,相當於perl/python語言的.*?,在匹配過程中,對於.*正則,一有匹配立即執行,而不是等.*搜索了所有字符再一一返回
在使用正則表達式時,我們應該盡量避免遞歸調用,遞歸容易導致堆棧溢出。比如:
/<table((?!<table).)*?</a>/isU 就會發生錯誤,而使用 /<table.*?</a>/i 就正常。
那麼如何增加win平台下 ThreadStackSize 的大小呢? 在apache的配置文件 httpd.conf 裡啟用 “Include conf/extra/httpd-mpm.conf”(刪除前面的注釋#),然後在 httpd-mpm.conf 文件裡的 mpm_winnt_module 配置模塊裡設置 “ThreadStackSize 8400000”即可(大約8M)。
<IfModule mpm_winnt_module>
ThreadStackSize 8400000
ThreadsPerChild 200
MaxRequestsPerChild 10000
Win32DisableAcceptEx
</IfModule>
這裡需要注意的是,32位的Apache程序只能最多使用大約2GB內存空間! 因此,ThreadStackSize 和ThreadsPerChild 的值相乘後(8M * 200)不應該超過2G,否則無法啟動apache,出現的錯誤日志如下:
[Thu Apr 11 20:02:45 2013] [crit] (OS 8)存儲空間不足,無法處理此命令。 : Child 4832: _beginthreadex failed. Unable to create all worker threads. Created 212 of the 220 threads requested with the ThreadsPerChild configuration directive.
通過上面的提示,飄易可以告訴大家的是在我的這台服務器上,當線程堆棧大小設為8M時,我可以設置的線程數最多是212個。