一.記憶體溢出解決方案
在做資料統計分析時,經常會遇到大數組,可能會發生記憶體溢出,這裡分享一下我的解決方案。還是用例子來說明這個問題,如下:
假定日誌中存放的記錄數為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); //銷毀$sunset($p);$mm=memory_get_usage();echo $p.'<br />';echo $m-$mm;?>
我們將$s和$p都使用unset()銷毀,這時再看記憶體佔用量之差也是272,說明這樣也可以釋放記憶體。那麼,我們可以得到另外一條結論:
結論二、只有當指向該變數的所有變數(如引用變數)都被銷毀後,才會釋放記憶體。
以上就是最全的php記憶體溢出的解決方案 的內容,更多相關內容請關注topic.alibabacloud.com(www.php.cn)!