Swoole4.x之协程变量访问安全与协程连接池实现

栏目: PHP · 发布时间: 5年前

内容简介:为什么说有访问安全问题呢?传统地,在php的的环境中,很少有Phper遇到所谓变量安全访问问题。举个例子,代码大约如下:这个是在fpm模式下,很常见的数据库连接单例模式的使用。乍一看没有问题,但实际上,在协程环境下,会出现连接跨协程使用问题,举例如下我们会发现,以上代码当中,协程2的数据污染到了协程1的数据,那么因此这样肯定是不行的。

访问安全问题

为什么说有访问安全问题呢?传统地,在 php 的的环境中,很少有Phper遇到所谓变量安全访问问题。举个例子,代码大约如下:

class db
{
    protected static $instance;
    protected $dbCon;
    
    function __construct()
    {
        /*
         * 我们这里用stdclass来模拟一个数据库连接
         */
        $this->dbCon = new \stdClass();
    }

    public static function getInstance()
    {
        if(!isset(self::$instance)){
            self::$instance = new db();
        }
        return self::$instance;
    }

    function dbCon()
    {
        return $this->dbCon;
    }
}

$con = db::getInstance()->dbCon();
$con->key = 'new';
var_dump($con->key);

这个是在fpm模式下,很常见的数据库连接单例模式的使用。乍一看没有问题,但实际上,在协程环境下,会出现连接跨协程使用问题,举例如下

go(function (){
    go(function (){
        db::getInstance()->dbCon()->key = 'one';
        //假设这 sql 执行了1s
        \co::sleep(1);
        var_dump(db::getInstance()->dbCon()->key);
    });
    go(function (){
        db::getInstance()->dbCon()->key = 'two';
        //假设这sql执行了0.1s
        \co::sleep(0.1);
        var_dump(db::getInstance()->dbCon()->key);
    });
});

我们会发现,以上代码当中,协程2的数据污染到了协程1的数据,那么因此这样肯定是不行的。

上下文管理器

为了解决这个问题,我们引入协程上下文管理这样的概念,由此来实现每个协程环境内的数据隔离。

class dbContext
{
    private $container = [];

    private static $instance;

    public static function getInstance()
    {
        if(!isset(self::$instance)){
            self::$instance = new dbContext();
        }
        return self::$instance;
    }

    function dbCon()
    {
        $cid = \co::getCid();
        if(!isset($this->container[$cid])){
            $this->container[$cid] = new stdClass();
            defer(function (){
                $this->destroy();
            });
        }
        return $this->container[$cid];
    }

    function destroy()
    {
        $cid = \co::getCid();
        if(!isset($this->container[$cid])){
            unset($this->container[$cid]);
        }
    }
}

go(function (){
    go(function (){
        dbContext::getInstance()->dbCon()->key = 'one';
        //假设这sql执行了1s
        \co::sleep(1);
        var_dump(dbContext::getInstance()->dbCon()->key);
    });
    go(function (){
        dbContext::getInstance()->dbCon()->key = 'two';
        //假设这sql执行了0.1s
        \co::sleep(0.1);
        var_dump(dbContext::getInstance()->dbCon()->key);
    });
});

以上代码中,我们用每个协程的id,来作为每个协程栈的数据token,用了defer方法,实现了每个协程退出的时候的数据自动清理,从而避免了内存泄露。

通用版本的连接池与协程上下文管理

我们不难发现,以上代码中,实际上依旧是短连接的管理方式,没办法对链接进行复用,由于本文章仅做基础原理讲解之用,具体有兴趣的同学,可以查看下Easyswoole这个框架的连接池和协程上下文管理器,项目主页在 www.easyswoole.com ,若觉得喜欢,有帮助,可以给easyswoole的github仓库点个赞。


以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

机器学习

机器学习

(美)Tom Mitchell / 曾华军、张银奎、等 / 机械工业出版社 / 2008-3 / 35.00元

《机器学习》展示了机器学习中核心的算法和理论,并阐明了算法的运行过程。《机器学习》综合了许多的研究成果,例如统计学、人工智能、哲学、信息论、生物学、认知科学、计算复杂性和控制论等,并以此来理解问题的背景、算法和其中的隐含假定。《机器学习》可作为计算机专业 本科生、研究生教材,也可作为相关领域研究人员、教师的参考书。一起来看看 《机器学习》 这本书的介绍吧!

HTML 压缩/解压工具
HTML 压缩/解压工具

在线压缩/解压 HTML 代码

CSS 压缩/解压工具
CSS 压缩/解压工具

在线压缩/解压 CSS 代码

HSV CMYK 转换工具
HSV CMYK 转换工具

HSV CMYK互换工具