生命周期

进程生命周期

  • 每个进程都有很长的生命周期
  • 每个进程是独立运行的互不干扰的
  • 每个进程在其生命周期内可以处理多个请求
  • 进程在收到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数组元素越来越多导致内存泄漏。

更多相关信息请参考 内存泄漏