多多色-多人伦交性欧美在线观看-多人伦精品一区二区三区视频-多色视频-免费黄色视屏网站-免费黄色在线

國內(nèi)最全IT社區(qū)平臺 聯(lián)系我們 | 收藏本站
阿里云優(yōu)惠2
您當(dāng)前位置:首頁 > php開源 > php教程 > win2003使用preg_match_all導(dǎo)致apache崩潰解決辦法

win2003使用preg_match_all導(dǎo)致apache崩潰解決辦法

來源:程序員人生   發(fā)布時間:2014-04-19 07:02:04 閱讀次數(shù):4019次

Apache/2.2.9 (Win32) + PHP/5.2.17,在使用正則表達式 preg_match_all (如 preg_match_all("/ni(.*?)wo/", $html, $matches);)進行分析匹配比較長的字符串 $html 時(大于10萬字節(jié),一般用于分析采集回來的網(wǎng)頁源碼),Apache服務(wù)器會崩潰自動重啟。

在Apache錯誤日志里有這樣的提示:

  1. [Thu Apr 11 18:31:31 2013] [notice] Parent: child process exited with status 128 -- Restarting. 
  2. [Thu Apr 11 18:31:31 2013] [notice] Apache/2.2.9 (Win32) PHP/5.2.17 configured -- resuming normal operations 
  3. [Thu Apr 11 18:31:31 2013] [notice] Server built: Jun 13 2008 04:04:59 
  4. [Thu Apr 11 18:31:31 2013] [notice] Parent: Created child process 2964 
  5. [Thu Apr 11 18:31:31 2013] [notice] Disabled use of AcceptEx() WinSock2 API 
  6. [Thu Apr 11 18:31:31 2013] [notice] Child 2964: Child process is running 
  7. [Thu Apr 11 18:31:31 2013] [notice] Child 2964: Acquired the start mutex. 
  8. [Thu Apr 11 18:31:31 2013] [notice] Child 2964: Starting 350 worker threads. 
  9. [Thu Apr 11 18:31:31 2013] [notice] Child 2964: Listening on port 80. 

經(jīng)過查閱Apache官方以及論壇資料后,發(fā)現(xiàn)win平臺下用正則 preg_match_all 或preg_match 分析比較長的字符串時,導(dǎo)致apache崩潰重啟的原因是windows平臺下默認分配的線程堆棧空間 ThreadStackSize 太小導(dǎo)致的。 win32默認只有256KB,而在 linux下默認值是 8M,這就是為什么同樣的程序在 linux平臺下正常,而在 win平臺下不正常的原因。

根據(jù)PCRE library的官方說明:256 KB 的堆棧空間對應(yīng)的pcre.recursion_limit大小應(yīng)該不超過524。

下面就是一張Stacksize和pcre.recursion_limit對應(yīng)的建議安全值,超過這個數(shù)值就極有可能發(fā)生堆棧溢出,apache crash:

  1. Stacksize   pcre.recursion_limit 
  2.  64 MB      134217 
  3.  32 MB      67108 
  4.  16 MB      33554 
  5.   8 MB      16777 
  6.   4 MB      8388 
  7.   2 MB      4194 
  8.   1 MB      2097 
  9. 512 KB      1048 
  10. 256 KB      524 

如果你沒有調(diào)整堆棧大小,就必須在使用正則的PHP頁面最開頭加入:

  1. <?php 
  2. ini_set("pcre.recursion_limit""524"); // PHP default is 100,000. 
  3. ?> 

查看具體的錯誤可以使用下面的代碼:

  1. $resultsArray = preg_match_all("/table.*?<a>/isU"$html$contents); 
  2. if ($resultsArray === 0){ 
  3. echo get_pcre_err(); 
  4. function get_pcre_err(){ 
  5.         $pcre_err = preg_last_error();  // PHP 5.2 and above. 
  6.         if ($pcre_err === PREG_NO_ERROR) { 
  7.             $msg = 'Successful non-match.'
  8.         } else { 
  9.             // preg_match error! 
  10.             switch ($pcre_err) { 
  11.                 case PREG_INTERNAL_ERROR: 
  12.                     $msg = 'PREG_INTERNAL_ERROR'
  13.                     break
  14.                 case PREG_BACKTRACK_LIMIT_ERROR: 
  15.                     $msg = 'PREG_BACKTRACK_LIMIT_ERROR'
  16.                     break
  17.                 case PREG_RECURSION_LIMIT_ERROR: 
  18.                     $msg = 'PREG_RECURSION_LIMIT_ERROR'
  19.                     break
  20.                 case PREG_BAD_UTF8_ERROR: 
  21.                     $msg = 'PREG_BAD_UTF8_ERROR'
  22.                     break
  23.                 case PREG_BAD_UTF8_OFFSET_ERROR: 
  24.                     $msg = 'PREG_BAD_UTF8_OFFSET_ERROR'
  25.                     break
  26.                 default
  27.                     $msg = 'Unrecognized PREG error'
  28.                     break
  29.             } 
  30.         } 
  31.     return($msg); 

對于正則的修飾符 isU 說明:

i: 表示in-casesensitive,即大小寫不敏感

s: PCRE_DOTALL,表示點號可以匹配換行符。

U: 表示PCRE_UNGREEDY,表示非貪婪,相當(dāng)于perl/python語言的.*?,在匹配過程中,對于.*正則,一有匹配立即執(zhí)行,而不是等.*搜索了所有字符再一一返回,在使用正則表達式時,我們應(yīng)該盡量避免遞歸調(diào)用,遞歸容易導(dǎo)致堆棧溢出。比如:

/<table((?!<table).)*?</a>/isU 就會發(fā)生錯誤,而使用 /<table.*?</a>/i 就正常。

那么如何增加win平臺下 ThreadStackSize 的大小呢? 在apache的配置文件 httpd.conf 里啟用 “Include conf/extra/httpd-mpm.conf”(刪除前面的注釋#),然后在 httpd-mpm.conf 文件里的 mpm_winnt_module 配置模塊里設(shè)置 “ThreadStackSize 8400000”即可(大約8M),代碼如下:

  1. <IfModule mpm_winnt_module> 
  2.     ThreadStackSize 8400000 
  3.     ThreadsPerChild      200 
  4.     MaxRequestsPerChild    10000 
  5.     Win32DisableAcceptEx 
  6. </IfModule> 

這里需要注意的是,32位的Apache程序只能最多使用大約2GB內(nèi)存空間! 因此,ThreadStackSize 和ThreadsPerChild 的值相乘后(8M * 200)不應(yīng)該超過2G,否則無法啟動apache,出現(xiàn)的錯誤日志如下:

  1. [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. 

通過上面的提示,飄易可以告訴大家的是在我的這臺服務(wù)器上,當(dāng)線程堆棧大小設(shè)為8M時,我可以設(shè)置的線程數(shù)最多是212個。

生活不易,碼農(nóng)辛苦
如果您覺得本網(wǎng)站對您的學(xué)習(xí)有所幫助,可以手機掃描二維碼進行捐贈
程序員人生
------分隔線----------------------------
分享到:
------分隔線----------------------------
關(guān)閉
程序員人生
主站蜘蛛池模板: 欧美一级aa免费毛片 | 毛片的网址 | 亚洲欧美精品日韩欧美 | 国产精品96久久久久久久 | 国产亚洲福利一区二区免费看 | 欧美日韩国产亚洲人成 | 欧美激情视频二区 | 男女晚上日日麻批视频不挡 | 国产精品久久一区 | 国产亚洲免费观看 | 免费一级大毛片a一观看不卡 | jizz欧美| 免费爱爱网站 | 国产日韩一区在线精品欧美玲 | 国产成人在线视频观看 | 国产成人一区二区三区免费观看 | 国产精品视_精品国产免费 国产精品视频1区 | 久久亚洲人成国产精品 | 尤物视频在线观看免费 | 成人亚洲在线观看 | 日本无卡αv免费视频 | 国产福利自产拍在线观看 | 国产一级爱片在线播放 | 国产午夜毛片v一区二区三区 | 夜趣第一宅男福社区国产 | 欧美一级成人一区二区三区 | jizz在亚洲| 欧美亚洲一区 | 一级做性色a爰片久久毛片 一级做性色a爰片久久毛片免费 | 波多野结衣视频免费 | 亚洲欧美一区二区三区另类 | a天堂中文在线官网 | 欧美一级毛片欧美一级成人毛片 | 91一区二区在线观看精品 | 久久精品这里是免费国产 | 欧美一级aa免费毛片 | 欧美高清性刺激毛片 | 精品国产一级毛片 | 免费播放春色aⅴ视频 | 18videosex性加拿大 | 美毛片|