生命週期
進程生命週期
- 每個進程都有很長的生命週期
- 每個進程是獨立運行的互不干擾的
- 每個進程在其生命週期內可以處理多個請求
- 進程在收到
stop
reload
restart
命令時會執行退出,結束本次生命週期
提示
每個進程都是獨立互不干擾的,這意味着每個進程都維護著自己的資源、變量和類實例等,表現在每個進程都有自己的數據庫連接,一些單例在每個進程初始化一次,那麼多個進程就會初始化多次。
請求生命週期
- 每個請求會產生一個
$request
對象 $request
對象在請求處理完畢後會被回收
控制器生命週期
- 每個控制器每個進程只會實例化一次,多個進程實例化多次(關閉控制器複用除外,參見控制器生命週期)
- 控制器實例會被當前進程內多個請求共享(關閉控制器複用除外)
- 控制器生命週期在進程退出後結束(關閉控制器複用除外)
關於變量生命週期
webman是基於php開發的,所以它完全遵循php的變量回收機制。業務邏輯裡產生的臨時變量包括 new 關鍵字創建的類的實例,在函數或者方法結束後自動回收,無需手動 unset
釋放。也就是說webman開發與傳統框架開發體驗基本一致。例如下面例子中 $foo
實例會隨著 index 方法執行完畢而自動釋放:
<?php
namespace app\controller;
use app\service\Foo;
use support\Request;
class IndexController
{
public function index(Request $request)
{
$foo = new Foo(); // 這裡假設有一個 Foo 類
return response($foo->sayHello());
}
}
如果你想某個類的實例被複用,則可以將類保存到類的靜態屬性中或長生命週期對象(如控制器)的屬性中,也可以使用Container容器的get方法來初始化類的實例,例如:
<?php
namespace app\controller;
use app\service\Foo;
use support\Container;
use support\Request;
class IndexController
{
public function index(Request $request)
{
$foo = Container::get(Foo::class);
return response($foo->sayHello());
}
}
Container::get()
方法用於創建並保存類的實例,下次再次以同樣的參數再次調用時將返回之前創建的類實例。
注意
Container::get()
只能初始化沒有構造參數的實例。Container::make()
可以創建帶構造函數參數的實例,但是與Container::get()
不同的是,Container::make()
並不會複用實例,也就是說即使以同樣的參數Container::make()
始終返回一個新的實例。
關於內存泄漏
絕大部分情況下,我們的業務代碼並不會發生內存泄漏(極少有用戶反饋發生內存泄漏),我們只要稍微注意下長生命週期的數組數據不要無限擴張即可。請看以下代碼:
<?php
namespace app\controller;
use support\Request;
class FooController
{
// 數組屬性
public $data = [];
public function index(Request $request)
{
$this->data[] = time();
return response('hello index');
}
public function hello(Request $request)
{
return response('hello webman');
}
}
控制器默認是長生命週期的(關閉控制器複用除外),同樣的控制器的 $data
數組屬性也是長周期的,隨著 foo/index
請求不斷增加,$data
數組元素越來越多導致內存泄漏。
更多相關信息請參考 內存泄漏