關(guān)于內(nèi)存泄漏
webman是常駐內(nèi)存框架,所以我們需要稍微關(guān)注下內(nèi)存泄漏的情況。不過開發(fā)者不必過于擔(dān)心,因為內(nèi)存泄漏發(fā)生在非常極端的條件下,而且很容易規(guī)避。webman開發(fā)與傳統(tǒng)框架開發(fā)體驗基本一致,不必為內(nèi)存管理做多余的操作。
提示
webman自帶的monitor進(jìn)程會監(jiān)控所有進(jìn)程內(nèi)存使用情況,如果進(jìn)程使用內(nèi)存即將達(dá)到php.ini里memory_limit
設(shè)定的值時,會自動安全重啟對應(yīng)的進(jìn)程,達(dá)到釋放內(nèi)存的作用,期間對業(yè)務(wù)沒有影響。
內(nèi)存泄漏定義
隨著請求的不斷增加,webman占用的內(nèi)存也無限增加(注意是無限增加),達(dá)到幾百M(fèi)甚至更多,這種是內(nèi)存泄漏。
如果是內(nèi)存有增長,但是后面不再增長不算內(nèi)存泄漏。
一般進(jìn)程占用幾十M內(nèi)存是很正常的情況,當(dāng)進(jìn)程處理超大請求或者維護(hù)海量連接時,單個進(jìn)程內(nèi)存占用可能會達(dá)到上百M(fèi)也是常有的事。這部分內(nèi)存使用后php可能并不會全部交還操作系統(tǒng)。而是留著復(fù)用,所以可能會出現(xiàn)處理某個大請求后內(nèi)存占用變大不釋放內(nèi)存的情況,這是正?,F(xiàn)象。(調(diào)用gc_mem_caches()方法可以釋放部分空閑內(nèi)存)
內(nèi)存泄漏是如何發(fā)生的
內(nèi)存泄漏發(fā)生必須滿足以下兩個條件:
- 存在長生命周期的數(shù)組(注意是長生命周期的數(shù)組,普通數(shù)組沒事)
- 并且這個長生命周期的數(shù)組會無限擴(kuò)張(業(yè)務(wù)無限向其插入數(shù)據(jù),從不清理數(shù)據(jù))
如果1 2條件同時滿足(注意是同時滿足),那么將會產(chǎn)生內(nèi)存泄漏。反之不滿足以上條件或者只滿足其中一個條件則不是內(nèi)存泄漏。
長生命周期的數(shù)組
webman里長生命周期的數(shù)組包括:
- static關(guān)鍵字的數(shù)組
- 單例的數(shù)組屬性
- global關(guān)鍵字的數(shù)組
注意
webman中允許使用長生命周期的數(shù)據(jù),但是需要保證數(shù)據(jù)內(nèi)的數(shù)據(jù)是有限的,元素個數(shù)不會無限擴(kuò)張。
以下分別舉例說明
無限膨脹的static數(shù)組
class Foo
{
public static $data = [];
public function index(Request $request)
{
self::$data[] = time();
return response('hello');
}
}
以static
關(guān)鍵字定義的$data
數(shù)組是長生命周期的數(shù)組,并且示例中$data
數(shù)組隨著請求不斷增加而不斷膨脹,導(dǎo)致內(nèi)存泄漏。
無限膨脹的單例數(shù)組屬性
class Cache
{
protected static $instance;
public $data = [];
public function instance()
{
if (!self::$instance) {
self::$instance = new self;
}
return self::$instance;
}
public function set($key, $value)
{
$this->data[$key] = $value;
}
}
調(diào)用代碼
class Foo
{
public function index(Request $request)
{
Cache::instance()->set(time(), time());
return response('hello');
}
}
Cache::instance()
返回一個Cache單例,它是一個長生命周期的類實例,雖然它的$data
屬性雖然沒有使用static
關(guān)鍵字,但是由于類本身是長生命周期,所以$data
也是長生命周期的數(shù)組。隨著不斷向$data
數(shù)組里添加不同key的數(shù)據(jù),程序占用內(nèi)存也月來越大,造成內(nèi)存泄漏。
注意
如果 Cache::instance()->set(key, value) 添加的key是有限數(shù)量的,則不會內(nèi)存泄漏,因為$data
數(shù)組并沒有無限膨脹。
無限膨脹的global數(shù)組
class Index
{
public function index(Request $request)
{
global $data;
$data[] = time();
return response($foo->sayHello());
}
}
global 關(guān)鍵字定義的數(shù)組并不會在函數(shù)或者類方法執(zhí)行完畢后回收,所以它是長生命周期的數(shù)組,以上代碼隨著請求不斷增加會產(chǎn)生內(nèi)存泄漏。同理在函數(shù)或者方法內(nèi)以static關(guān)鍵字定義的數(shù)組也是長生命周期的數(shù)組,如果數(shù)組無限膨脹也會內(nèi)存泄漏,例如:
class Index
{
public function index(Request $request)
{
static $data = [];
$data[] = time();
return response($foo->sayHello());
}
}
建議
建議開發(fā)者不用特別關(guān)注內(nèi)存泄漏,因為它極少發(fā)生,如果不幸發(fā)生我們可以通過壓測找到哪段代碼產(chǎn)生泄漏,從而定位出問題。即使開發(fā)者沒有找到泄漏點(diǎn),webman自帶的monitor服務(wù)會適時安全重啟發(fā)生內(nèi)存泄漏的進(jìn)程,釋放內(nèi)存。
如果你實在想盡量規(guī)避內(nèi)存泄漏,可以參考以下建議。
- 盡量不使用
global
,static
關(guān)鍵字的數(shù)組,如果使用確保其不會無限膨脹 - 對于不熟悉的類,盡量不使用單例,用new關(guān)鍵字初始化。如果需要單例,則查看其是否有無限膨脹的數(shù)組屬性