easyswoole数据库连接池_如何在 Swoole 中优雅的实现 MySQL 连接池

如何在 Swoole 中优雅的实现 MySQL 连接池

一、为什么需要连接池 ?

数据库连接池指的是程序和数据库之间保持一定数量的连接不断开,

并且各个请求的连接可以相互复用,

减少重复连接数据库带来的资源消耗,

一定程度上提高了程序的并发性能。

二、连接池实现要点

协程:使用 MySQL 协程客户端。

使用 MySQL 协程客户端,是为了能在一个 Worker 阻塞的时候,

让出 CPU 时间片去处理其他的请求,提高整个 Worker 的并发能力。

连接池存储介质:使用 \swoole\coroutine\channel 通道。

使用 channel 能够设置等待时间,等待其他的请求释放连接。

并且在等待期间,同样也可以让出 CPU 时间片去处理其他的请求。

假设选择 array 或 splqueue,无法等待其他的请求释放连接。

那么在高并发下的场景下,可能会出现连接池为空的现象。

如果连接池为空了,那么 pop 就直接返回 null 了,导致连接不可用。

注:因此不建议选择 array 或 splqueue。

三、连接池的具体实现

use Swoole\Coroutine\Channel;

use Swoole\Coroutine\MySQL;

class MysqlPool

{

private $min; // 最小连接数

private $max; // 最大连接数

private $count; // 当前连接数

private $connections; // 连接池

protected $freeTime; // 用于空闲连接回收判断

public static $instance;

/**

* MysqlPool constructor.

*/

public function __construct()

{

$this->min = 10;

$this->max = 100;

$this->freeTime = 10 * 3600;

$this->connections = new Channel($this->max + 1);

}

/**

* @return MysqlPool

*/

public static function getInstance()

{

if (is_null(self::$instance)) {

self::$instance = new self();

}

return self::$instance;

}

/**

* 创建连接

* @return MySQL

*/

protected function createConnection()

{

$conn = new MySQL();

$conn->connect([

'host' => 'mysql',

'port' => '3306',

'user' => 'root',

'password' => 'root',

'database' => 'fastadmin',

'timeout' => 5

]);

return $conn;

}

/**

* 创建连接对象

* @return array|null

*/

protected function createConnObject()

{

$conn = $this->createConnection();

return $conn ? ['last_used_time' => time(), 'conn' => $conn] : null;

}

/**

* 初始化连接

* @return $this

*/

public function init()

{

for ($i = 0; $i < $this->min; $i++) {

$obj = $this->createConnObject();

$this->count++;

$this->connections->push($obj);

}

return $this;

}

/**

* 获取连接

* @param int $timeout

* @return mixed

*/

public function getConn($timeout = 3)

{

if ($this->connections->isEmpty()) {

if ($this->count < $this->max) {

$this->count++;

$obj = $this->createConnObject();

} else {

$obj = $this->connections->pop($timeout);

}

} else {

$obj = $this->connections->pop($timeout);

}

return $obj['conn']->connected ? $obj['conn'] : $this->getConn();

}

/**

* 回收连接

* @param $conn

*/

public function recycle($conn)

{

if ($conn->connected) {

$this->connections->push(['last_used_time' => time(), 'conn' => $conn]);

}

}

/**

* 回收空闲连接

*/

public function recycleFreeConnection()

{

// 每 2 分钟检测一下空闲连接

swoole_timer_tick(2 * 60 * 1000, function () {

if ($this->connections->length() < intval($this->max * 0.5)) {

// 请求连接数还比较多,暂时不回收空闲连接

return;

}

while (true) {

if ($this->connections->isEmpty()) {

break;

}

$connObj = $this->connections->pop(0.001);

$nowTime = time();

$lastUsedTime = $connObj['last_used_time'];

// 当前连接数大于最小的连接数,并且回收掉空闲的连接

if ($this->count > $this->min && ($nowTime - $lastUsedTime > $this->freeTime)) {

$connObj['conn']->close();

$this->count--;

} else {

$this->connections->push($connObj);

}

}

});

}

}

$httpServer = new swoole_http_server('127.0.0.1',9501);

$httpServer->set(['work_num' => 1]);

$httpServer->on('WorkerStart', function ($request, $response) {

MysqlPool::getInstance()->init()->recycleFreeConnection();

});

$httpServer->on('Request', function ($request, $response){

$conn = MysqlPool::getInstance()->getConn();

$conn->query('SELECT * FROM fa_admin WHERE id=1');

MysqlPool::getInstance()->recycle($conn);

});

$httpServer->start();

四、总结

定时维护空闲连接到最小值。

使用用完数据库连接之后,需要手动回收连接到连接池。

使用 channel 作为连接池的存储介质。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/358010.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CVE-2022-22965:Spring Framework远程代码执行漏洞

CVE-2022-22965&#xff1a;Spring Framework远程代码执行漏洞 本文仅为验证漏洞&#xff0c;在本地环境测试验证&#xff0c;无其它目的 漏洞编号&#xff1a; CVE-2022-22965 漏洞说明&#xff1a; Spring framework 是Spring 里面的一个基础开源框架&#xff0c;其目的是…

js中四种创建对象的方式

一、 1 var user new Object(); 2 user.first"Brad"; 3 user.last"Dayley"; 4 user.getName function( ) { return this.first " " this.last; } 二、 1 var user { 2 first: Brad, 3 last: Dayley, 4 getName: function( ) { return…

CSS系列讲解-总目录

总目录: 欢迎来到孙叫兽的《CSS系列讲解》,蓝色字体为传送门,点击进入即可。本专栏已完结,大前端专栏支持更新。 玩转CSS系列: 什么是CSS?你真的理解? CSS页面DEMO CSS基本语法? 如何玩转CSS的Id 和 Class选择器? 怎么玩转CSS内部样式表与外部样式表? 怎么样才…

红队信息收集自动化工具-水泽(ShuiZe)

红队信息收集自动化工具-水泽&#xff08;ShuiZe&#xff09; 文章目录 红队信息收集自动化工具-水泽&#xff08;ShuiZe&#xff09;0x01 介绍0x02 安装0x03 效果展示0x04 POC编写0x05 使用方法0x06 实现原理Web -> 存活探测0x07 项目地址 0x01 介绍 定位&#xff1a;协助…

注释不好吗?

前几天&#xff0c;我在有关Spring XML与注释的文章中运用了自己的原则&#xff0c;轻松进入了这个主题。 对于我目前正在编写此新应用程序的团队来说&#xff0c;这种简单的输入方式也是我不会使事情复杂化的方式&#xff0c;该应用程序的生产寿命可能为3-5年&#xff08;如果…

前端工程师应该达到什么水平,找工作薪资才比较高?

当然是水平越高&#xff0c;越容易找到工作&#xff0c;薪资越高在竞争这么激烈的2020年&#xff0c;就需要更加的努力&#xff0c;充实自己&#xff0c;让自己不被代替&#xff01;两条路&#xff1a;自学或者找培训班&#xff0c;找培训班的话&#xff0c;我推荐达内和传智播…

【转】使用JMeter对数据库做压力测试

作为一名开发人员&#xff0c;大多情况下都会认真的做好功能测试&#xff0c;但是却常常忽略了软件开发之后的压力测试&#xff0c;尤其是在面向大量用户同时使用的Web应用系统的开发过程&#xff0c;压力测试往往是不够充分的。近期我在一个求职招聘型的网站项目中就对压力测试…

python中的大数据品牌运营专业公司_国内最好的专业数据分析公司有哪些?

说说我知道的几家&#xff0c;都是在各自领域最好的。大数据平台星环&#xff0c;做Hadoop生态系列的大数据底层平台公司。也是国内唯一入选过Gartner魔力象限的大数据平台公司。Hadoop是开源的&#xff0c;星环主要做的是把Hadoop不稳定的部分优化&#xff0c;功能细化&#x…

POC以及day下载链接地址

POC以及day下载链接地址 https://github.com/Threekiii/Awesome-POC https://github.com/coffeehb/Some-PoC-oR-ExP https://github.com/PeiQi0/PeiQi-WIKI-Book https://github.com/luck-ying/Library-POC https://github.com/helloexp/0day https://github.com/BaizeSec/byl…

孙叫兽进阶之路之Gitlab的使用(图文教程)

简介&#xff1a; GitLab是一个利用 Ruby on Rails 开发的开源应用程序&#xff0c;实现一个自托管的Git项目仓库&#xff0c;可通过Web界面进行访问公开的或者私人项目。 它拥有与Github类似的功能&#xff0c;能够浏览源代码&#xff0c;管理缺陷和注释。可以管理团队对仓库的…

PyQt4(使用ui)

1.使用qt designer设计界面&#xff0c;保存为test1.ui&#xff1a; 2.使用pyuic4 test1.ui -o ui.py生成ui代码。 3.程序载入。 import sys import ui from PyQt4 import QtCore, QtGuiclass MyWidget( QtGui.QWidget ):def __init__(self):super(MyWidget, self).__init__() …

docker升级步骤及注意事项

centos系统默认安装的docker版本是1.13版本&#xff0c;在安装部分镜像时可能出现兼容问题&#xff0c;本文通过实际操作总结Docker升级最新版本步骤及可能出现的问题&#xff0c;供各位参考。 环境&#xff1a;CentOS Linux release 7.6.1810 (Core) docker升级操作&#xf…

孙叫兽进阶之路之源代码配置管理过程(图文教程)

简介:配置管理(Configuration Management,CM)是通过技术或行政手段对软件产品及其开发过程和生命周期进行控制、规范的一系列措施。配置管理的目标是记录软件产品的演化过程,确保软件开发者在软件生命周期中各个阶段都能得到精确的产品配置。

进击的Objective-C--------Objective-C基础(-)

1.面向过程和面向对象(面向对象三大特性:封装 继承 多态)面向对象编程:分析解决问题组成的对象,从中抽象出类,调用方法(协调对象间的联系与通信),解决问题.面向过程编程:分析解决问题的步骤,实现函数,一次调用2类和对象:类和对象是面向对象的核心类:具有相同特征和行为的事物的…

ivy maven_将Maven与Ivy集成

ivy maven问题是&#xff1a;您在Ivy存储库中&#xff08;只有那里&#xff09;有一些资源&#xff0c;您想在基于Maven的项目中使用这些资源。 可能的解决方案&#xff1a; 由于Ivy可以轻松使用Maven样式的存储库&#xff08;因此&#xff0c;您的Ivy客户端可以继续使用Ivy并进…

video 微信 标签层级过高_什么是微信小程序二级分销系统?如何玩转?

微信二级分销系统是通过帮助企业打造微分销商城&#xff0c;从店铺、商品、会员、分销、营销、数据分析等不同功能模块&#xff0c;让一个微信店铺焕发无限可能。微分销系统基于二级分销&#xff0c;以全员开店&#xff0c;以客推客模式迅速推动销量增长&#xff0c;快速招募微…

docker搭建简单的ctf题目

0x01 docker常用命令 1.拉取镜像。 docker pull [image] 2.查看docker当前镜像。 docker image ls 或 docker images 3.新建一个docker容器&#xff0c;并映射端口号。 docker run -d -p [host port]:[docker port] [image] 4.查看运行中的docker容器。 docker ps -a 5.进入一…

使用爱思助手备份苹果手机数据的方法

背景:前段时间刚给对象买的一个紫色的苹果11,128G的那种,最近发现电池电量忽然就少很多,电池除了点问题,去苹果售后店准备换一个电池,还在保修期,区分一下售后店(回厂修十多天)及专卖店(有备用电池)。今天提前备份一下数据,防止数据丢失,一般内存不大可以使用手机…

高级Java教程

课程大纲 学习Java基础很容易。 但是&#xff0c;真正钻研该语言并研究其更高级的概念和细微差别将使您成为一名出色的Java开发人员。 网络上充斥着“软”&#xff0c;“便宜”&#xff0c;“低端” Java教程&#xff0c;但是所缺少的实际上是将您带入新的高度的材料。 本课程…

Cool!15个超炫的 CSS3 文本特效【上篇】

每一个网页设计师都希望创建出让用户能够赏识的网站。当然&#xff0c;这是不可能满足每个人的口味的。幸运的是&#xff0c;我们有最强大的工具和资源。实际上&#xff0c;我们非常多的网站模板&#xff0c;框架&#xff0c;内容管理系统&#xff0c;先进的工具和其他的资源可…