有的時候我們需要處理大量的數據,過多的循環可能導致記憶體耗盡,該如何解決呢?下面小編給大家解決PHP里大量資料循環時記憶體耗盡的問題,需要的朋友可以參考下。希望對大家有幫助。
最近在開發一個PHP程式的時候遇到以下問題:
PHP Fatal error: Allowed memory size of 268 435 456 bytes exhausted
錯誤訊息顯示允許的最大記憶體已經耗盡。遇到這樣的錯誤起初讓我很詔異,但轉眼一想,也不奇怪,因為我正在開發的這個程式是要用一個foreach循環語句在一個有4萬條記錄的表裡全表搜索具有特定特徵的數據,也就是說,一次要把4萬個數據取出,然後逐一檢查每天數據。可想而知,4萬個資料全部載入到記憶體中,記憶體不爆才怪。

畢竟程式設計這麼多年,我隱約記得PHP裡提供有非一次全部載入資料的API,是像處理串流媒體一樣,隨用隨取隨丟、數據並不會累積在記憶體的查詢方法。經過簡單的搜索,果然在官方網站上找到的正確的用法。
這個問題在PHP的官方網站上叫緩衝查詢和非緩衝查詢(Buffered and Unbuffered queries)。 PHP的查詢缺省模式是緩衝模式。也就是說,查詢資料結果會一次全部提取到記憶體裡供PHP程式處理。這樣給了PHP程式額外的功能,比如說,計算行數,將 指標指向某一行等。更重要的是程式可以對資料集反覆進行二次查詢和過濾等操作。但這種緩衝查詢模式的缺陷就是消耗內存,也就是用空間換速度。
相對的,另外一種PHP查詢模式是非緩衝查詢,資料庫伺服器會一條一條的返回數據,而不是一次全部返回,這樣的結果就是PHP程式消耗較少的內存,但卻增加了資料庫伺服器的壓力,因為資料庫會一直等待PHP來取數據,一直到資料全部取完。
很顯然,緩衝查詢模式適用於小資料量查詢,而非緩衝查詢適應於大資料量查詢。
對於PHP的緩衝模式查詢大家都知道,以下列舉的範例是如何執行非緩衝查詢API。
非緩衝查詢方法一: mysqli
#
1 2 3 4 5 6 7 8 9 10 | <?php
$mysqli = new mysqli( "localhost" , "my_user" , "my_password" , "world" );
$uresult = $mysqli ->query( "SELECT Name FROM City" , MYSQLI_USE_RESULT);
if ( $uresult ) {
while ( $row = $uresult ->fetch_assoc()) {
echo $row ['Name'] . PHP_EOL;
}
}
$uresult ->close();
?>
|
登入後複製
非緩衝查詢方法二: pdo_mysql
#
1 2 3 4 5 6 7 8 9 10 11 | <a href= "https://www.php.cn/php-weizijiaocheng-339749.html" target= "_blank" style= "font-size: 16px; text-decoration: none;" ><?php
$pdo = new PDO( "mysql:host=localhost;dbname=world" , 'my_user', 'my_pass');
$pdo ->setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, false);
$uresult = $pdo ->query( "SELECT Name FROM City" );
if ( $uresult ) {
while ( $row = $uresult ->fetch(PDO::FETCH_ASSOC)) {
echo $row ['Name'] . PHP_EOL;
}
}
?></a>
|
#####################################################################非緩衝查詢方法三: mysql##################
1 2 3 4 5 6 7 8 9 10 11 | <?php
$conn = mysql_connect( "localhost" , "my_user" , "my_pass" );
$db = mysql_select_db( "world" );
$uresult = mysql_unbuffered_query( "SELECT Name FROM City" );
if ( $uresult ) {
while ( $row = mysql_fetch_assoc( $uresult )) {
echo $row ['Name'] . PHP_EOL;
}
}
?>
|
登入後複製
##############相關推薦:###### #########php 迴圈讀取檔案內容###################PHP 迴圈時間控制緩衝方法_PHP教學######## ##########PHP 迴圈語句基本語法結構筆記_PHP教學##########
以上是PHP大量循環時的記憶體最佳化實例的詳細內容。更多資訊請關注PHP中文網其他相關文章!