生命週期

進程生命週期

  • 每個進程都有很長的生命週期
  • 每個進程是獨立運行的互不干擾的
  • 每個進程在其生命週期內可以處理多個請求
  • 當進程收到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數組元素越來越多導致內存泄漏。

更多相關信息請參考 內存泄漏