1. 程式人生 > >PHP指令碼佔用記憶體太多,解決方案

PHP指令碼佔用記憶體太多,解決方案

Fatal Error: Allowed memory size of xxxxxx bytes exhausted

今天遇到伺服器很多自動任務的php指令碼佔用服務記憶體過多,並且程式不在運行了。

解決方法:

在執行PHP程式,通常會遇到“Fatal Error: Allowed memory size of xxxxxx bytes exhausted”的錯誤, 這個意味著PHP指令碼使用了過多的記憶體,並超出了系統對其設定的允許最大記憶體。解決這個問題,首先需要檢視你的程式是否分配了過多的記憶體,在程式沒有問題的情況下,你可以通過一下方法來增加PHP的記憶體限制(memory_limit)。

檢查php的記憶體限制值

為了檢視這個值,你需要建立一個空的php檔案,比如view-php-info.php。然後將一下程式碼貼到裡面。

<?php phpinfo(); ?>

將這個指令碼放到你的Web伺服器上,然後在瀏覽器中呼叫它。這時你可以看到你的PHP環境配置的資訊,其中有一部分是關於“memory_limit”的, 如下圖:

注:你可以用這種方法來檢視php的其他引數設定,不僅僅是memory_limit

memory_limit應該設為多少?

這個完全依賴於你的應用的要求。比如Wordpress,執行起核心程式碼需要32MB。Drupal 6則要求這個值最小為16MB,並推薦設定為32MB。如果你又安裝不少的外掛(plugins),尤其是那些要進行影象處理的模組,那麼你可能需要128MB或更高的記憶體。

如何設定memory_limit

方法1: php.ini

最簡單或常用的方法是修改php.ini

1.首先找到對你的網站生效的php.ini檔案 由於有多個地方都可以設定php的引數,找到正確的配置檔案,並進行更改是首先要做的一步。如果你上面的方法建立了php檔案來檢視其配置引數,則你可以找到“Loaded Configuration File”這一項,以下是個例子:

對於Linux使用者,你可以通過執行“php -i | grep Loaded Configuration File”來找到對應的配置檔案。而Windows使用者,你可以嘗試修改你的php安裝目錄下的php.ini。

2.編輯php.ini 在php.ini中,找到“memory_limit”這一項,如果沒有,你可以在檔案的尾部自己增加這個引數。以下是一些設定範例

memory_limit = 128M ; 可以將128M改為任何你想設定的值

儲存檔案

3.重啟web 伺服器 如果是web伺服器使用Apache, 則執行:

httpd restart

有些情況下,你可能不被允許私修改php.ini。比如如果你購買了虛擬主機服務,但是你的服務商確禁止你修改這個檔案。那麼,你可以需要考慮用其他方法來增加memory_limit的值。



那麼我們要怎麼從程式的根本上來解決這個問題呢,請往下看:


PHP開發過程中,高手都會用unset來釋放記憶體,開始也沒有深入瞭解,就跟著用,當然不是所有變數都unset,因為開發這麼久了,也沒感覺不unset有問題,但是最近的一個程式頻繁出問題,總是記憶體不夠用,為是麼我知道記憶體不夠用呢,因為使用ini_set(‘memory_limit’, ’2000M’);增加記憶體以後,程式就正常了,所以很明顯,記憶體不夠用,但是PHP不是會釋放記憶體的嗎,為是麼會導致記憶體不夠呢,沒錯,PHP是會釋放記憶體,但是是每個函式執行完釋放一次,整個程式跑完才能完全釋放,而為了採集,每個程式都由許多函式來跑,這些函式有的使用記憶體過多,還沒能等到執行完畢釋放記憶體,就死掉了,整個程式也就中斷了,下面看幾個例子,會有更深的體會。

    function test()
    {
    echo memory_get_usage().”    test函式開始使用記憶體\n”;

    $a[]=’a';
    unset($a);
    echo memory_get_usage().”    給陣列a賦值後使用記憶體\n”;

    $b[]=’b';
    unset($b);
    echo memory_get_usage().”    給陣列b賦值後使用記憶體\n”;

    $c[]=’c';
    unset($c);
    echo memory_get_usage().”    給陣列c賦值後使用記憶體\n”;
    }

    echo memory_get_usage() . ”    PHP讀入當前檔案所用記憶體,程式碼越多,記憶體佔用越多\n”;
    test();
    echo memory_get_usage() . ”    執行完test函式所用記憶體\n”;
    test();
    echo memory_get_usage() . ”    再執行一次test函式所用記憶體\n”;
    ?>

結果:

321124    PHP讀入當前檔案所用記憶體,程式碼越多,記憶體佔用越多

321144    test函式開始使用記憶體

321144    給陣列a賦值後使用記憶體

321144    給陣列b賦值後使用記憶體

321144    給陣列c賦值後使用記憶體

321144    執行完test函式所用記憶體

321144    test函式開始使用記憶體

321144    給陣列a賦值後使用記憶體

321144    給陣列b賦值後使用記憶體

321144    給陣列c賦值後使用記憶體

321144    再執行一次test函式所用記憶體

    function test()
    {
    echo memory_get_usage().”    test函式開始使用記憶體\n”;

    $a[]=’a';
    //unset($a);
    echo memory_get_usage().”    給陣列a賦值後使用記憶體\n”;

    $b[]=’b';
    //unset($b);
    echo memory_get_usage().”    給陣列b賦值後使用記憶體\n”;

    $c[]=’c';
    //unset($c);
    echo memory_get_usage().”    給陣列c賦值後使用記憶體\n”;
    }

    echo memory_get_usage() . ”    PHP讀入當前檔案所用記憶體,程式碼越多,記憶體佔用越多\n”;
    test();
    echo memory_get_usage() . ”    執行完test函式所用記憶體\n”;
    test();
    echo memory_get_usage() . ”    再執行一次test函式所用記憶體\n”;
    ?>

結果:

320896    PHP讀入當前檔案所用記憶體,程式碼越多,記憶體佔用越多

320916    test函式開始使用記憶體

321120    給陣列a賦值後使用記憶體

321324    給陣列b賦值後使用記憶體

321528    給陣列c賦值後使用記憶體

320916    執行完test函式所用記憶體

320916    test函式開始使用記憶體

321120    給陣列a賦值後使用記憶體

321324    給陣列b賦值後使用記憶體

321528    給陣列c賦值後使用記憶體

320916    再執行一次test函式所用記憶體

從以上結果得知:

1.函式內不unset變數的話記憶體會越用越多,但是這個佔用是一時的,這個函式一執行完這些記憶體就會釋放,所以不會導致整個程式佔用記憶體過多,這就需要保證一個函式不能使用記憶體過多,否則超過額定記憶體(PHP的額定記憶體一般設定為38M,可以通過ini_set(‘memory_limit’, ’2000M’)設定),程式就會退出。

2.函式內unset了變數,對於整個程式來說佔用的記憶體更多,不unset反而用得少,這是是麼原因呢,我個人覺得,從巨集觀角度來說,我們不去觀察函式,而是把函式看成是一個操作,那這個操作內容越多,佔用的記憶體就越多,所以執行unset這個操作後,使用的記憶體反而更多,當然,這個記憶體的增加是可以忽略不計的,而函式內部的變數如果很大的話,不unset的話,可能執行這個函式的時候就卡死在那裡了,所以unset還是必須的

3.最後查了很多資料,瞭解到,在PHP4年代,PHP執行是記憶體不夠用再向系統要,用完了再給回系統,所以記憶體佔用從工作管理員裡面可以很清楚的看到,但是PHP5之後,PHP執行的時候會先向系統徵用一塊大記憶體,然後自己管理,超過這個記憶體程式再向系統要,所以在工作管理員是看不出程式執行時記憶體使用情況的,這也導致很多人覺得unset可用可不用
---------------------
作者:Sunface撩技術
來源:CSDN
原文:https://blog.csdn.net/erlib/article/details/38488825
版權宣告:本文為博主原創文章,轉載請附上博文連結!