熱點推薦:
您现在的位置: 電腦知識網 >> 編程 >> PHP編程 >> 正文

關於PHP內存溢出問題的解決方法

2013-11-15 12:29:18  來源: PHP編程 

  一.內存溢出解決方案
在做數據統計分析時經常會遇到大數組可能會發生內存溢出這裡分享一下我的解決方案還是用例子來說明這個問題如下
假定日志中存放的記錄數為那麼解決方案如下

復制代碼 代碼如下:
ini_set(‘memory_limitM); //重置php可以使用的內存大小為M一般在遠程主機上是不能修改phpini文件的只能通過程序設置在safe_mode(安全模式)下ini_set失效
set_time_limit();//設置超時限制為6分鐘
$farr = $Uarr = $Marr = $IParr = $data = $_sub = array();
$spt = ”$@#!$”;
$root = ”/Data/webapps/VisitLog”;
$path = $dpath = $fpath = NULL;
$path = $root”/”date(“Ym”$timestamp);
$dpath = $path”/”date(“md”$timestamp);
for($j=;$j<;$j++){
$v = ($j < ) ? ”$j : $j;
$gpath = $dpath”/”$vphp”;
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 ); //每次取出$farr中
for($i=$scount=count($_sub);$i<$scount;$i++){
$arr = explode($spt$_sub[$i]);
$Uarr[] = $arr[]; //vurl
$Marr[] = $arr[]; //vmark
$IParr[] = $arr[]” |$nbsp;”$arr[]; //IP
}
unset($_sub);//用完及時銷毀
}
unset($farr);

  
這裡不難看出一方面我們要增加PHP可用內存大小另一方面只要我們想辦法對數組進行分批處理分而治之將用過的變量及時銷毀(unset)一般是不會出現溢出問題的

另外為了節省PHP程序內存損耗我們應當盡可能減少靜態變量的使用在需要數據重用時可以考慮使用引用(&)再一點就是數據庫操作完成後要馬上關閉連接一個對象使用完要及時調用析構函數(__destruct())

  二.unset銷毀變量並釋放內存問題
PHP的unset()函數用來清除銷毀變量不用的變量我們可以用unset()將它銷毀但是某些時候用unset()卻無法達到銷毀變 量占用的內存!我們先看一個例子

復制代碼 代碼如下:
<?php
$s=str_repeat(); //產生由組成的字符串
$m=memory_get_usage(); //獲取當前占用內存
unset($s);
$mm=memory_get_usage(); //unset()後再查看當前占用內存
echo $m$mm;
?>

  
最 後輸出unset()之前占用內存減去unset()之後占用內存如果是正數那麼說明unset($s)已經將$s從內存中銷毀(或者 說unset()之後內存占用減少了)可是我在PHP和windows平台下得到的結果是這是否可以說明unset($s)並沒有起 到銷毀變量$s所占用內存的作用呢?我們再作下面的例子

復制代碼 代碼如下:
<?php
$s=str_repeat(); //產生由組成的字符串
$m=memory_get_usage(); //獲取當前占用內存
unset($s);
$mm=memory_get_usage(); //unset()後再查看當前占用內存
echo $m$mm;
?>

  
這個例子和上面的例子幾乎相同唯一的不同是$s由組成即比第一個例子多了一個得到結果是這是否可以說 明unset($s)已經將$s所占用的內存銷毀了?
通過上面兩個例子我們可以得出以下結論
結論一unset()函數只能在變量值占用內存空間超過字節時才會釋放內存空間
那麼是不是只要變量值超過使用unset就可以釋放內存空間呢?我們再通過一個例子來測試一下

復制代碼 代碼如下:
<?php
$s=str_repeat(); //這和第二個例子完全相同
$p=&$s;
$m=memory_get_usage();
unset($s); //銷毀$s
$mm=memory_get_usage();
echo $p<br />;
echo $m$mm;
?>

  
刷新頁面我們看到第一行有第二行是按理說我們已經銷毀了$s而$p只是引用$s的變量應該是沒有內容了另 外unset($s)前後內存占用沒變化!現在我們再做以下的例子

復制代碼 代碼如下:
<?php
$s=str_repeat(); //這和第二個例子完全相同
$p=&$s;
$m=memory_get_usage();
$s=null; //設置$s為null
$mm=memory_get_usage();
echo $p<br />;
echo $m$mm;
?>

  
現在刷新頁面我們看到輸出$p已經是沒有內容了unset()前後內存占用量之差是即已經清除了變量占用的內存本例中的$s=null也 可以換成unset()如下

復制代碼 代碼如下:
<?php
$s=str_repeat(); //這和第二個例子完全相同
$p=&$s;
$m=memory_get_usage();
unset($s); //銷毀$s
unset($p);
$mm=memory_get_usage();
echo $p<br />;
echo $m$mm;
?>

  
我們將$s和$p都使用unset()銷毀這時再看內存占用量之差也是說明這樣也可以釋放內存那麼我們可以得到另外一條結論
結論二只有當指向該變量的所有變量(如引用變量)都被銷毀後才會釋放內存


From:http://tw.wingwit.com/Article/program/PHP/201311/21047.html
    推薦文章
    Copyright © 2005-2013 電腦知識網 Computer Knowledge   All rights reserved.