龙空技术网

PHP进阶教程-phper需要知道常驻内存中的上下文是什么东西

IT不是挨踢 36

前言:

目前我们对“php常驻内存框架”大致比较关注,姐妹们都需要剖析一些“php常驻内存框架”的相关知识。那么小编在网上收集了一些有关“php常驻内存框架””的相关内容,希望我们能喜欢,兄弟们快快来学习一下吧!

​我们在讲进程、线程的时候多次提及了上下文切换,上下文其实在我们生活中也经常遇到,可能只是我们没有细心留意。

什么是上下文?

上小学考试的时候,经常我们试卷上会有一段文章的句子,让我们解释出那句话的前因后果;我们也学过承上启下,上下文跟这个差不多,根据文章前面的内容和后面的内容得知这句话的意思。不然给你一段话,你根本不知道这段话出现在文章的这个地方表示什么。

例如:我和她假期去旅游。

这句话我们可以引出三个问题。她是指的谁?假期是哪个假期?去哪里旅游?贸然的给我们一句话我们不知道这段话说的意思。因为这段话没有上文也没有下文,我们不知道说的她是谁;还有这个假期是过去式还是将来时;如果这句话改一改,把上下文带进来。

下面是小明和小强的对白:

小明:小强不好意思啊。我假期没空和你一起加班了。

小强:为什么啊?

小明:我和她假期去旅游。

小强:和谁去哪呢?

小明:小红啊,她说五一有空,我和她一起出去三亚下海。

小强:滚吧,重色轻友的混蛋。

这样就有了上下文,我们就知道这个她是谁,假期是什么时候,目的地是哪里。这就是语文里面理解的上下文的左右。

上下文没有具体的解释,也没有某一个具体的实例来解释,上下文会根据应用程序的生命周期而变化。也就是说同样的一句话在不同的上下文环境中,所表示的意思不一样。

为什么使用上下文?

我们理解了什么是上下文,那么我们说说编程里面的上下文,说白了编程里面的上下文就是一个环境。跟前面了解的情况一下,没有上下文我们不知道某一个变量处于某一个时刻或者某一进程,线程代表什么意思。因此上下文就是告诉这些变量所处的位置,例如一个进程切换的时候需要保存进程的状态,资源保护好现场,等到再次切换回来的时候从中断的地方恢复。

如图所示,我们在每一个请求都有变量、数据库资源、文件资源等等。在不同的上下文环境中,这些请求的值所表示不一样的内容。例如在请求1中userid是1002,在请求2中userid是999。

在使用常驻内存框架的时候很多时候为了性能,控制器会弄成单例的,所以我们在使用的时候不能像以往在fpm模式下开发一样,使用基类来继承属性在控制器使用共同属性;例如我们在中间件做授权验证,如果想要把这个用户id继续传到控制器使用,怎么办呢?我们又不能通过基类的方式来继承,这时候可以把这个id挂载到上下文中。

最直观的理解就是每一次请求都会有一个上下文来管理这个请求的所有产生的内容,做到每一个请求的变量和资源达到隔离状态,避免了不同的请求会出现数据污染的情况。

为什么在php-fpm不需要上下文?

其实在php-fpm模式开发我们经常不需要管理上下文,因为在php-fpm模式开发不是常驻内存的,并且php-fpm帮我们管理了上下文,不需要开发者自己管理,而在常驻内存开发中需要维护每个请求的环境。

常驻内存没有上下文隔离产生的问题

常驻内存下采用单例的控制器,不会因为每次请求都new一次控制器。

BaseController.php

class BaseController{    protected $num = 0;}

Controller.php

class Controller extends BaseController{    public function index(){        $this->num++;        return "index:".$this->num;    }    public function test(){        $this->num++;        return "test:".$this->num;    }}

Server.php

include './BaseController.php';include './Controller.php';class Server{    private $server;    private $controller;    public function __construct(){        $this->controller = new Controller();        $this->server = new Swoole\Http\Server("127.0.0.1", 9501);        $this->server->on('request', function ($request, $response){            $c = $request->get['c']??'index';            $res = call_user_func([$this->controller,$c]);            $response->end($res);        });        $this->server->start();    }}new Server();

分别请求一下:

实战:手写swoole框架上下文管理

通过上下文隔离每一个请求的信息,避免数据污染。

Context.php

use Swoole\Coroutine;class Context{    private static $context = [];    public static function get($key = null){        $tid = Coroutine::getCid();        if ($key == null){            return self::$context[$tid] ?? null;        }else{            return self::$context[$tid][$key] ?? null;        }    }    /**     * Set context     * @param $key     * @param $context     */    public static function set($key,$context){        $tid = Coroutine::getCid();        self::$context[$tid][$key] = $context;    }    public static function destroy(){        $tid = Coroutine::getCid();        unset(self::$context[$tid]);    }    public static function printContext(){        print_r(self::$context);    }}

Controller.php

class Controller{    public function index(){        $num = Context::get("num");        $num++;        return "index:".$num;    }    public function test(){        $num = Context::get("num");        $num++;        return "test:".$num;    }}

Server.php

include './Context.php';include './Controller.php';class Server{    private $server;    private $controller;    public function __construct(){        $this->controller = new Controller();        $this->server = new Swoole\Http\Server("127.0.0.1", 9501);        $this->server->on('request', function ($request, $response){            defer(function(){                Context::destroy();            });            Context::set('num',0);            Context::printContext();            $c = $request->get['c']??'index';            $res = call_user_func([$this->controller,$c]);            $response->end($res);        });        $this->server->start();    }}new Server();

可以通过协程id获取当前请求的上下文环境,并且可以在任意地方只要得到协程id就可以获取上下文,可以把一些和当前请求关联的数据通过上下文传递。

标签: #php常驻内存框架