一.記憶體溢出解決方案
在做資料統計分析時,常常會遇到大數組,可能會發生記憶體溢出,這裡分享我的解決方案。還是用範例來說明這個問題,如下:
假定日誌中存放的記錄數為500000條,那麼解如下:
複製程式碼如下:
ini_set(‘memory_limit','64M'); //重置php可以使用的内存大小为64M,一般在远程主机上是不能修改php.ini文件的,只能通过程序设置。注:在safe_mode(安全模式)下,ini_set失效 set_time_limit(600);//设置超时限制为6分钟 $farr = $Uarr = $Marr = $IParr = $data = $_sub = array(); $spt = ”$@#!$”; $root = ”/Data/webapps/VisitLog”; $path = $dpath = $fpath = NULL; $path = $root.”/”.date(“Y-m”,$timestamp); $dpath = $path.”/”.date(“m-d”,$timestamp); for($j=0;$j<24;$j++){ $v = ($j < 10) ? ”0″.$j : $j; $gpath = $dpath.”/”.$v.”.php”; if(!file_exists($gpath)){ continue; } else { $arr = file($gpath);////将文件读入数组中 array_shift($arr);//移出第一个单元-》<?php exit;?> $farr = array_merge($farr,$arr); unset($arr); } } if(empty($this->farr)){ echo ”<p><center>没有相关记录!</center></p>”; exit; } while(!empty($farr)){ $_sub = array_splice($farr, 0, 10000); //每次取出$farr中1000个 for($i=0,$scount=count($_sub);$i<$scount;$i++){ $arr = explode($spt,$_sub[$i]); $Uarr[] = $arr[1]; //vurl $Marr[] = $arr[2]; //vmark $IParr[] = $arr[3].” |$nbsp;”.$arr[1]; //IP } unset($_sub);//用完及时销毁 } unset($farr);
這裡,不難看出,一方面,我們要增加PHP可用記憶體大小,另一方面,只要我們想辦法對數組進行分批處理,分而治之,將用過的變數及時銷毀(unset),一般是不會出現溢出問題的。
另外,為了節省PHP程式記憶體損耗,我們應盡可能減少靜態變數的使用,在需要資料重用時,可以考慮使用參考(&)。再一點就是:資料庫操作完成後,要馬上關閉連線;一個物件使用完,要及時呼叫析構函數(__destruct())。
二. unset銷毀變數並釋放記憶體問題
PHP的unset()函數用來清除、銷毀變量,不用的變量,我們可以用unset()將它銷毀。但某些時候,用unset()卻無法達到銷毀變 量所佔用的記憶體!我們先看一個例子:
複製代碼代碼如下:
<?php $s=str_repeat('1',255); //产生由255个1组成的字符串 $m=memory_get_usage(); //获取当前占用内存 unset($s); $mm=memory_get_usage(); //unset()后再查看当前占用内存 echo $m-$mm; ?>
最後輸出unset()之前佔用內存減去unset()之後佔用內存,如果是正數,那麼說明unset($s)已經將$s從記憶體中銷毀(或者說,unset()之後記憶體佔用減少了),可是我在PHP5和windows平台下,得到的結果是:0。這是否可以說明,unset($s)並沒有起 到銷毀變數$s所佔用記憶體的作用?我們再作下面的例子:
複製程式碼如下:
<?php $s=str_repeat('1',256); //产生由256个1组成的字符串 $m=memory_get_usage(); //获取当前占用内存 unset($s); $mm=memory_get_usage(); //unset()后再查看当前占用内存 echo $m-$mm; ?>
這個例子,和上面的例子幾乎相同,唯一的不同是,$s由256個1組成,即比第一個例子多了一個1,得到結果是:272。這是否可以說 明,unset($s)已經將$s所佔用的記憶體銷毀了?
透過上面兩個例子,我們可以得到以下結論:
結論一、unset()函數只能在變數值佔用記憶體空間超過256位元組時才會釋放記憶體空間。
那麼是不是只要變數值超過256,使用unset就可以釋放記憶體空間呢?我們再用一個例子來測試一下:
複製程式碼如下:
<?php $s=str_repeat('1',256); //这和第二个例子完全相同 $p=&$s; $m=memory_get_usage(); unset($s); //销毁$s $mm=memory_get_usage(); echo $p.'<br />'; echo $m-$mm; ?>
刷新頁面,我們看到第一行有256個1,第二行是0,照理說我們已經銷毀了$s,而$p只是引用$s的變量,應該是沒有內容了,另外,unset($s)前後內存佔用沒變化!現在我們再做以下的例子:
複製程式碼如下:
<?php $s=str_repeat('1',256); //这和第二个例子完全相同 $p=&$s; $m=memory_get_usage(); $s=null; //设置$s为null $mm=memory_get_usage(); echo $p.'<br />'; echo $m-$mm; ?>
現在刷新頁面,我們看到,輸出$p已經是沒有內容了,unset( )前後記憶體佔用量之差是272,也就是已經清除了變數佔用的記憶體。本例的$s=null也可以換成unset(),如下:
複製程式碼如下:
<?php $s=str_repeat('1',256); //这和第二个例子完全相同 $p=&$s; $m=memory_get_usage(); unset($s); //销毁$s unset($p); $mm=memory_get_usage(); echo $p.'<br />'; echo $m-$mm; ?>
我們將$s和$p都使用unset()銷毀,這時再看記憶體佔用量之差也是272,說明這樣也可以釋放記憶體。那麼,我們可以得到另外一條結論:
結論二、只有當指向該變數的所有變數(如引用變數)都被銷毀後,才會釋放記憶體。
以上就是最全的php記憶體溢出的解決方法 的內容,更多相關內容請關注PHP中文網(www.php.cn)!