亚洲精品成人_精品成人一区_999视频在线播放_免费黄色在线_亚洲成人久久久_久久www免费视频

通用連接池

EasySwoole 實現的通用的協程連接池管理。

組件要求

  • php: >=7.1.0
  • ext-json: *
  • easyswoole/component: ^2.2.1
  • easyswoole/spl: ^1.3
  • easyswoole/utility: ^1.1

安裝方法

composer require easyswoole/pool

倉庫地址

easyswoole/pool

池配置

在實例化一個連接池對象時,需要傳入一個連接池配置對象 EasySwoole\Pool\Config,該對象的屬性如下:

配置項 默認值 說明 備注
$intervalCheckTime 15 * 1000 定時器執行頻率(毫秒),默認值為 15 s 用于定時執行連接池對象回收,創建操作
$maxIdleTime 10 連接池對象最大閑置時間(秒) 超過這個時間未使用的對象將會被定時器回收
$maxObjectNum 20 連接池最大數量 每個進程最多會創建 $maxObjectNum 個連接池對象,如果對象都在使用,則會返回空,或者等待連接空閑
$minObjectNum 5 連接池最小數量(熱啟動) 當連接池對象總數低于 $minObjectNum 時,會自動創建連接,保持連接的活躍性,讓控制器能夠盡快地獲取連接
$getObjectTimeout 3.0 獲取連接池中連接對象的超時時間 當連接池為空時,會等待 $getObjectTimeout 秒,如果期間有連接空閑,則會返回連接對象,否則返回 null
$extraConf 額外配置信息 在實例化連接池前,可以把一些額外配置放到這里,例如數據庫配置信息、redis 配置等等
$loadAverageTime 0.001 負載閾值 并發來臨時,連接池內對象達到 maxObjectNum,此時并未達到 intervalCheckTime 周期檢測,因此設定了一個 5s 負載檢測,當 5s 內,取出總時間/取出連接總次數,會得到一個平均取出時間,如果小于此閾值,說明此次并發峰值非持續性,將回收 5% 的連接

池管理器

池管理器可以做全局的連接池管理,例如在 EasySwooleEvent.php 中的 initialize 事件中注冊,然后可以在控制器中獲取連接池然后進行獲取連接:

下面以使用實現 easyswoole/redis 組件實現 Redis 連接池為例:

前提:先使用 composer 安裝 easyswoole/redis 組件:

composer require easyswoole/redis

定義 RedisPool 管理器

基于 AbstractPool 實現:

新增文件 \App\Pool\RedisPool.php,內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link http://www.b3f21.cn
 * @document http://www.b3f21.cn
 * @contact http://www.b3f21.cn/Preface/contact.html
 * @license https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */

namespace App\Pool;

use EasySwoole\Pool\AbstractPool;
use EasySwoole\Pool\Config;
use EasySwoole\Redis\Config\RedisConfig;
use EasySwoole\Redis\Redis;

class RedisPool extends AbstractPool
{
    protected $redisConfig;

    /**
     * 重寫構造函數,為了傳入 redis 配置
     * RedisPool constructor.
     * @param Config      $conf
     * @param RedisConfig $redisConfig
     * @throws \EasySwoole\Pool\Exception\Exception
     */
    public function __construct(Config $conf, RedisConfig $redisConfig)
    {
        parent::__construct($conf);
        $this->redisConfig = $redisConfig;
    }

    protected function createObject()
    {
        // 根據傳入的 redis 配置進行 new 一個 redis 連接
        $redis = new Redis($this->redisConfig);
        return $redis;
    }
}

或者基于 MagicPool 實現:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link http://www.b3f21.cn
 * @document http://www.b3f21.cn
 * @contact http://www.b3f21.cn/Preface/contact.html
 * @license https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */

namespace App\Pool;

use EasySwoole\Pool\Config;
use EasySwoole\Pool\MagicPool;
use EasySwoole\Redis\Config\RedisConfig;
use EasySwoole\Redis\Redis;

class RedisPool1 extends MagicPool
{
    /**
     * 重寫構造函數,為了傳入 redis 配置
     * RedisPool constructor.
     * @param Config $config 連接池配置
     * @param RedisConfig $redisConfig
     * @throws \EasySwoole\Pool\Exception\Exception
     */
    public function __construct(Config $config, RedisConfig $redisConfig)
    {
        parent::__construct(function () use ($redisConfig) {
            $redis = new Redis($redisConfig);
            return $redis;
        }, $config);
    }
}

不管是基于 AbstractPool 實現還是基于 MagicPool 實現效果是一致的。

注冊連接池管理對象

EasySwooleEvent.php 中的 initialize/mainServerCreate 事件中注冊,然后可以在控制器中獲取連接池然后進行獲取連接:

<?php

public static function initialize()
{
    // TODO: Implement initialize() method.
    date_default_timezone_set('Asia/Shanghai');

    $config = new \EasySwoole\Pool\Config();

    $redisConfig1 = new \EasySwoole\Redis\Config\RedisConfig(Config::getInstance()->getConf('REDIS1'));
    $redisConfig2 = new \EasySwoole\Redis\Config\RedisConfig(Config::getInstance()->getConf('REDIS2'));

    // 注冊連接池管理對象
    \EasySwoole\Pool\Manager::getInstance()->register(new \App\Pool\RedisPool($config,$redisConfig1), 'redis1');
    \EasySwoole\Pool\Manager::getInstance()->register(new \App\Pool\RedisPool($config,$redisConfig2), 'redis2');
}

在控制器中獲取連接池中連接對象,進行調用:

<?php

public function index()
{
    // 取出連接池管理對象,然后獲取連接對象(getObject)
    $redis1 = \EasySwoole\Pool\Manager::getInstance()->get('redis1')->getObj();
    $redis2 = \EasySwoole\Pool\Manager::getInstance()->get('redis2')->getObj();

    $redis1->set('name', '仙士可');
    var_dump($redis1->get('name'));

    $redis2->set('name', '仙士可2號');
    var_dump($redis2->get('name'));

    // 回收連接對象(將連接對象重新歸還到連接池,方便后續使用)
        \EasySwoole\Pool\Manager::getInstance()->get('redis1')->recycleObj($redis1);
        \EasySwoole\Pool\Manager::getInstance()->get('redis2')->recycleObj($redis2);

    // 釋放連接對象(將連接對象直接徹底釋放,后續不再使用)
    // \EasySwoole\Pool\Manager::getInstance()->get('redis1')->unsetObj($redis1);
    // \EasySwoole\Pool\Manager::getInstance()->get('redis2')->unsetObj($redis2);
}

池對象方法

方法名稱 參數 說明 備注
createObject 抽象方法,創建連接對象
recycleObj $obj 回收一個連接
getObj float $timeout = null, int $tryTimes = 3 在指定的超時時間 $timeout (秒)內獲取一個連接,會重復嘗試獲取 $tryTimes 次直到獲取到,獲取失敗則返回 null
unsetObj $obj 直接釋放一個連接
idleCheck int $idleTime 回收超過 $idleTime 未出隊使用的連接
itemIntervalCheck ObjectInterface $item 判斷當前客戶端是否還可用
intervalCheck 回收連接,以及熱啟動方法,允許外部調用熱啟動
keepMin ?int $num = null 保持最小連接(熱啟動)
getConfig 獲取連接池的配置信息
status 獲取連接池狀態信息 獲取當前連接池已創建、已使用、最大創建、最小創建數據
isPoolObject $obj 查看 $obj 對象是否由該連接池創建
isInPool $obj 獲取當前連接是否在連接池內未使用
destroy 銷毀該連接池
reset 重置該連接池
invoke callable $call,float $timeout = null 獲取一個連接,傳入到 $call 回調函數中進行處理,回調結束后自動回收連接
defer float $timeout = null 獲取一個連接,協程結束后自動回收

getObj

獲取一個連接池的對象:

<?php

go(function () {
    $redisPool = new \App\Pool\RedisPool(new \EasySwoole\Pool\Config(), new \EasySwoole\Redis\Config\RedisConfig(\EasySwoole\EasySwoole\Config::getInstance()->getConf('REDIS')));
    $redis = $redisPool->getObj();
    var_dump($redis->echo('仙士可'));
    $redisPool->recycleObj($redis);
});

通過 getObj 方法獲取的對象,都必須調用 recycleObj 或者 unsetObj 方法進行回收,否則連接池對象會越來越少。

unsetObj

直接釋放一個連接池的連接對象,其他協程不能再獲取到這個連接對象,而是會重新創建一個連接對象

釋放之后,并不會立即銷毀該對象,而是會在作用域結束之后銷毀

recycleObj

回收一個連接對象,回收之后,其他協程可以正常獲取這個連接對象。

回收之后,其他協程可以正常獲取這個連接,但在此時,該連接還處于當前協程中,如果再次調用該連接進行數據操作,將會造成協程混亂,所以需要開發人員自行約束,當對這個連接對象進行 recycleObj 操作后不能再操作這個對象

invoke

獲取一個連接,傳入到 $call 回調函數中進行處理,回調結束后自動回收連接:

<?php

go(function () {
    $redisPool = new \App\Pool\RedisPool(new \EasySwoole\Pool\Config(), new \EasySwoole\Redis\Config\RedisConfig(\EasySwoole\EasySwoole\Config::getInstance()->getConf('REDIS')));
    $redisPool->invoke(function (\EasySwoole\Redis\Redis $redis) {
        var_dump($redis->echo('仙士可'));
    });
});

通過該方法無需手動回收連接,在回調函數結束后,則自動回收

defer

獲取一個連接,協程結束后自動回收

<?php

go(function () {
    $redisPool = new \App\Pool\RedisPool(new \EasySwoole\Pool\Config(), new \EasySwoole\Redis\Config\RedisConfig(\EasySwoole\EasySwoole\Config::getInstance()->getConf('REDIS')));
    $redis = $redisPool->defer();
    var_dump($redis->echo('仙士可'));
});

通過該方法無需手動回收連接,在協程結束后,則自動回收

需要注意的事,defer 方法是協程結束后才回收,如果你當前協程運行時間過長,則會一直無法回收,直到協程結束

keepMin

保持最小連接(熱啟動)。

由于 easyswoole/pool 當剛啟動服務,出現過大的并發時,可能會突然需要幾十個甚至上百個連接,這時為了讓創建連接的時間分散,可以通過調用 keepMin 方法進行預熱啟動連接。

調用此方法后,將會預先創建 N 個連接,用于服務啟動之后的控制器直接獲取連接:

預熱使用示例如下:

EasySwooleEvent.php 中的 mainServerCreate 中,當 Worker 進程啟動后,熱啟動連接:

<?php

public static function mainServerCreate(EventRegister $register)
{
    $register->add($register::onWorkerStart, function (\swoole_server $server, int $workerId) {
        if ($server->taskworker == false) {
            //每個worker進程都預創建連接
            \EasySwoole\Pool\Manager::getInstance()->get('redis')->keepMin(10);
            var_dump(\EasySwoole\Pool\Manager::getInstance()->get('redis')->status());
        }
    });
}

將會輸出:

array(4) {
  ["created"]=>
  int(10)
  ["inuse"]=>
  int(0)
  ["max"]=>
  int(20)
  ["min"]=>
  int(5)
}

keepMin 是根據不同進程,創建不同的連接的,比如你有 10Worker 進程,將會輸出 10 次,總共創建 10 * 10 = 100 個連接

getConfig

獲取連接池的配置:

<?php

$redisPool = new \App\Pool\RedisPool(new \EasySwoole\Pool\Config(), new \EasySwoole\Redis\Config\RedisConfig(\EasySwoole\EasySwoole\Config::getInstance()->getConf('REDIS')));
var_dump($redisPool->getConfig());    

destroy

銷毀連接池。

調用之后,連接池剩余的所有鏈接都會被執行 unsetObj,并且將關閉連接隊列,調用之后 getObj 等方法都將失效:

<?php

go(function () {
    $redisPool = new \App\Pool\RedisPool(new \EasySwoole\Pool\Config(), new \EasySwoole\Redis\Config\RedisConfig(\EasySwoole\EasySwoole\Config::getInstance()->getConf('REDIS')));
    var_dump($redisPool->getObj());
    $redisPool->destroy();
    var_dump($redisPool->getObj());
});

reset

重置連接池。

調用 reset 之后,會自動調用 destroy 銷毀連接池,并在下一次 getObj 時重新初始化該連接池:

<?php

go(function (){
    $redisPool = new \App\Pool\RedisPool(new \EasySwoole\Pool\Config(), new \EasySwoole\Redis\Config\RedisConfig(\EasySwoole\EasySwoole\Config::getInstance()->getConf('REDIS')));
    var_dump($redisPool->getObj());
    $redisPool->reset();
    var_dump($redisPool->getObj());
});

status

獲取連接池當前狀態,調用之后將輸出:

<?php

go(function () {
    $redisPool = new \App\Pool\RedisPool(new \EasySwoole\Pool\Config(), new \EasySwoole\Redis\Config\RedisConfig(\EasySwoole\EasySwoole\Config::getInstance()->getConf('REDIS')));
    var_dump($redisPool->status());
});
array(4) {
  ["created"]=>
  int(10)
  ["inuse"]=>
  int(0)
  ["max"]=>
  int(20)
  ["min"]=>
  int(5)
}

idleCheck

回收空閑超時的連接

intervalCheck

調用此方法后,將調用 idleCheckkeepMin 方法,用于手動回收空閑連接和手動熱啟動連接

<?php

public function intervalCheck()
{
    $this->idleCheck($this->getConfig()->getMaxIdleTime());
    $this->keepMin($this->getConfig()->getMinObjectNum());
}

itemIntervalCheck

在內部定時器丟棄超時客戶端(閑置了超過指定時間,就先斷開)時,會觸發 itemIntervalCheck 函數,并將客戶端傳入,用戶通過這個函數可以實現判斷客戶端是否可用的邏輯。

該函數如果返回 true 代表可用(默認情況),返回false 將會導致該客戶端直接被丟棄。

可用于:維持客戶端心跳等。如 orm 中對其使用場景如下:維持 mysql 連接,減少 mysql 掉線 gone away 的幾率

<?php
/**
 * @param MysqliClient $item
 * @return bool
 */
public function itemIntervalCheck($item): bool
{
    /*
     * 如果最后一次使用時間超過 autoPing 間隔
     */
    /** @var Config $config */
    $config = $this->getConfig();
    if ($config->getAutoPing() > 0 && (time() - $item->__lastUseTime > $config->getAutoPing())) {
        try {
            // 執行一個sql觸發活躍信息
            $item->rawQuery('select 1');
            // 標記使用時間,避免被再次 gc
            $item->__lastUseTime = time();
            return true;
        } catch (\Throwable $throwable) {
            // 異常說明該鏈接出錯了,return 進行回收
            return false;
        }
    } else {
        return true;
    }
}

基本使用

定義池對象

<?php

class Std implements \EasySwoole\Pool\ObjectInterface
{
    function gc()
    {
        /*
         * 本對象被 pool 執行 unset 的時候
         */
    }

    function objectRestore()
    {
        /*
         * 回歸到連接池的時候
         */
    }

    function beforeUse(): ?bool
    {
        /*
         * 取出連接池的時候,若返回false,則當前對象被棄用回收
         */
        return true;
    }

    public function who()
    {
        return spl_object_id($this);
    }
}

定義池

<?php

class StdPool extends \EasySwoole\Pool\AbstractPool
{
    protected function createObject()
    {
        return new Std();
    }
}

不一定非要在創建對象方法 createObject() 中返回 EasySwoole\Pool\ObjectInterface 對象,任意類型對象均可

pool 組件版本 >= 1.0.2 后,提供了 魔術池 支持,可以快速進行定義池。

<?php

use \EasySwoole\Pool\MagicPool;

$magic = new MagicPool(function () {
    return new \stdClass(); // 示例,可以返回實現了 ObjectInterface 的對象
});

// 注冊后獲取
$test = $magic->getObj();
// 歸還
$magic->recycleObj($test);

魔術池構造方法的第二個參數,可以接收一個 configEasySwoole\Pool\Config 類),用于定義池數量等配置。

簡單示例

<?php

$config = new \EasySwoole\Pool\Config();
$pool = new StdPool($config);

go(function () use ($pool) {
    $obj = $pool->getObj();
    $obj2 = $pool->getObj();
    var_dump($obj->who());
    var_dump($obj2->who());
});

進階使用

基于 pool 實現的 Redis 連接池

[基于 pool 實現的 MySql 連接池]()

相關倉庫

easyswoole/redis-pool

亚洲精品成人_精品成人一区_999视频在线播放_免费黄色在线_亚洲成人久久久_久久www免费视频
  • <kbd id="eqi2k"><code id="eqi2k"></code></kbd><cite id="eqi2k"><tbody id="eqi2k"></tbody></cite>
    欧美一级二级三级| 国产视频不卡| 亚洲一区二区精品在线观看| 天堂资源在线亚洲资源| 中文字幕一区二区三区乱码| 欧美性色综合| 久久久青草婷婷精品综合日韩| www.久久爱.cn| 亚洲v欧美v另类v综合v日韩v| 欧美日韩天堂| 国产高清在线一区| 亚洲图片都市激情| 日韩视频三区| 蜜桃视频在线观看91| 亚洲网址在线| 懂色中文一区二区三区在线视频 | 国产精品国产三级国产专区53 | 亚洲bbw性色大片| 亚洲理论在线| 麻豆一区区三区四区产品精品蜜桃| 午夜一区二区三区| 欧美亚洲一区二区三区| 麻豆精品传媒视频| 99视频精品免费观看| 国产视频99| 在线一区日本视频| 日韩精品电影网站| 久久成人精品| 欧美精品导航| 韩国一区二区三区美女美女秀| 欧美另类亚洲| 精品国产一区二区三区四区精华| 黄色工厂这里只有精品| 国产精成人品localhost| 在线看视频不卡| www久久99| 亚洲精品一区二区三区蜜桃久 | 欧美日韩一区二区三区免费| 国产欧美激情| 亚洲视频在线观看日本a| 99久久无色码| 亚洲精品乱码久久久久久蜜桃91| 欧美一区2区三区4区公司二百| 一区二区精品| 欧美高清视频一区| 日本一区高清在线视频| 国产v亚洲v天堂无码| 最新日韩在线| 国产精品扒开腿做爽爽爽软件| 久久亚洲国产精品日日av夜夜| 3d动漫啪啪精品一区二区免费| 亚洲高清不卡| 欧美日韩一视频区二区| 色一情一乱一伦一区二区三区丨 | 一区视频二区视频| 精品乱码一区| 91久久国产综合久久蜜月精品| 伊人久久婷婷| 黄色工厂这里只有精品| 一区在线电影| 亚洲永久一区二区三区在线| 久久久精品有限公司| 91入口在线观看| 欧美亚洲三级| 久久综合中文色婷婷| 国产色综合网| 夜夜精品视频| 国产日韩一区二区三区| 亚洲国产影院| 亚洲色诱最新| 国产精品免费看| 午夜亚洲福利在线老司机| 999亚洲国产精| 亚洲精品1234| 性伦欧美刺激片在线观看| 日韩午夜一区| 午夜在线精品偷拍| 91原创国产| 久久精品一区二区三区不卡免费视频| 国产精品制服诱惑| 欧美精品123| 午夜视频久久久| 欧美日韩mv| 亚洲激情女人| 麻豆av一区二区三区| 99久久精品久久久久久ai换脸| 99久久精品久久久久久ai换脸| 成人av免费看| 欧美日韩国产综合在线| 一区二区视频在线播放| 影音先锋久久| 亚洲欧美日韩精品久久久 | 国产一区二区三区四区三区四| 国产精品porn| 久久狠狠久久综合桃花| 国产女主播一区二区| 日本不卡免费新一二三区| 欧美1区2区| 美日韩免费视频| 欧美精品v日韩精品v国产精品| 一区二区在线高清视频| 99香蕉国产精品偷在线观看 | 成人资源av| 视频一区二区三区免费观看| 欧美私人啪啪vps| 国产一区二区三区的电影| 99re视频在线播放| 性欧美.com| 国产一区二区三区高清| 久久精品magnetxturnbtih| 一区不卡视频| 久久精品一区二区国产| 日韩和欧美的一区二区| 中文精品一区二区三区| 精品欧美一区二区久久久伦 | 精品国产一区二区三区久久久久久| 久久精品aaaaaa毛片| 欧美色综合网| 国产在线精品二区| 国产精品国产三级国产专区53| 久久婷婷亚洲| 午夜欧美精品| 99爱精品视频| 欧美激情视频一区二区三区免费| 久久激情久久| 欧美日韩免费观看一区| 成人黄视频免费| 国产综合精品一区| 鲁鲁狠狠狠7777一区二区| 一本久道综合久久精品| 日韩福利一区二区三区| 亚洲一卡久久| 欧美三区不卡| 免费一区二区三区在在线视频| 99精品国产福利在线观看免费| 欧美日韩在线精品| 国产精品久久久一区二区三区| 一级做a爰片久久| 国产a一区二区| 99在线精品免费视频九九视| 日韩欧美亚洲v片| 国产一区二区三区无遮挡| 一本色道久久综合亚洲精品不| 亚洲激情图片| 久久精品日产第一区二区三区乱码 | 亚洲激情亚洲| 伊人久久青草| 麻豆av一区二区三区久久| 久久久精品网| 在线视频日韩| 亚洲精品123区| 欧美精品亚洲| 一区二区免费在线观看| 久久99国产精品| 国产精品久久久久久久久久久久午夜片| 久久天天综合| 亚洲精品乱码久久久久久蜜桃麻豆 | 伊人久久大香线蕉综合75| 久久99欧美| 久久99精品久久久水蜜桃| 国产精品毛片一区视频| 久久综合图片| 成人av片网址| 国产精品v欧美精品v日韩| 免费视频一区| 先锋亚洲精品| 亚洲久久成人| 国产欧美日韩综合一区在线观看| 欧美视频二区| 亚洲激情二区| 国产日韩高清一区二区三区在线| 亚洲一二区在线| 亚洲黄色三级| 国产欧美日韩综合精品二区| 99国产精品| 久久久青草婷婷精品综合日韩| 香蕉久久夜色精品| 亚洲欧美视频一区二区三区| 蜜桃伊人久久| 国模一区二区三区私拍视频| 蜜桃在线一区二区三区精品| 欧美日韩一区二| 欧美一区激情| 亚洲黄色一区| 久久人人精品| 久久狠狠久久综合桃花| 日韩高清在线播放| 亚洲无吗在线| 久久久久久婷| 欧美日韩在线播放一区二区| 亚洲不卡1区| 尹人成人综合网| 91观看网站| 日韩av电影免费观看| 国内精品久久久久久久97牛牛| 99日韩精品| 精品麻豆av| 自拍偷拍一区二区三区| 亚洲精品中文字幕在线| 国产精品国产精品|