数据库连接池实现

news/2025/1/11 10:11:41/文章来源:https://www.cnblogs.com/xingzhuz/p/18440521

欢迎访问的另一个博客: https://xingzhu.top/
源码链接: https://github.com/xingzhuz/MysqlLinkPool

前置知识:
相关的环境配置: https://xingzhu.top/archives/shu-ju-ku-lian-jie-chi-huan-jing-pei-zhi
MySQL API: https://subingwen.cn/mysql/mysql-api/
Jsoncpp API: https://xingzhu.top/archives/jsoncpp
C++多线程: https://xingzhu.top/archives/duo-xian-cheng-xian-cheng-chi

概述

  • 数据库连接池的作用是管理和复用数据库连接,以提高应用程序的性能和资源利用率
  • 通过预先创建一定数量的连接并将它们存储在池中,应用程序可以快速获取连接,而无需每次都进行昂贵的连接创建和销毁过程
  • 这不仅减少了延迟,还能有效控制数据库连接的数量,防止资源耗尽,从而提升整体系统的稳定性和响应速度

Pasted image 20240929162257

具体实现

  • 连接池只需要一个实例,所以连接池类应该是一个单例模式的类
ConnectionPool *ConnectionPool::getConnectPool()
{// 创建静态布局变量,访问范围就只有这个函数作用域,但是生命周期是到程序结束// 第一次调用,会在创建内存地址,但是第二次就直接使用这个地址了,不会再次创建,就返回之前创建的实例了// 这里使用的是单例模式中的懒汉模式,调用时创建static ConnectionPool pool;return &pool;
}
  • 所有的数据库连接应该维护到一个队列中,使用队列的目的是方便连接的添加和删除
  • 由于队列是连接池共享资源,需要使用互斥锁来保护队列数据的读写
  • 由于数据库有连接上限,过多会压力过大,导致性能降低,所以需要设置一个最大连接上限;为了应对突然的高并发操作,需要设置一个最小连接数,这个最小连接数是维护队列的最小数量,保证有这么多连接供使用
  • 客户端在满足条件的情况下,从连接池取连接,然后从进行使用,使用完毕后,归还这个连接,让这个连接重新加入队列中,不是销毁掉
  • 因此队列存储的是一个数据库连接的对象,即 MYSQL* ,为了便于使用,先封装一个 MYSQL 类,用于连接

封装数据库头文件

#pragma once
#include <iostream>
#include <mysql.h>
#include <chrono>
using namespace std;
using namespace chrono;
class MysqlConn
{
public:// 初始化数据库连接MysqlConn();// 释放数据库连接~MysqlConn();// 连接数据库bool connect(string user, string passwd, string dbName, string ip, unsigned short port = 3306);// 更新数据库: insert, update, deletebool update(string sql);// 查询数据库 select 语句bool query(string sql);// 遍历查询得到的结果集bool next();// 得到结果集中的字段值string value(int index);// 事务操作bool transaction();// 提交事务bool commit();// 事务回滚bool rollback();// 刷新起始的空闲时间点void refreshAliveTime();// 计算连接存活的总时长long long getAliveTime();private:void freeResult();                    // 释放结果集内存MYSQL *m_conn = nullptr;              // 数据库对象MYSQL_RES *m_result = nullptr;        // select 查询后的结果集MYSQL_ROW m_row = nullptr;            // 用于遍历结果集,保存一行数据steady_clock::time_point m_alivetime; // 创建绝对时钟
};

连接池头文件

#pragma once
#include <queue>
#include <mutex>
#include <condition_variable>
#include <thread>
#include "MysqlConn.h"
#include <atomic>
using namespace std;class ConnectionPool
{
public:// 设置静态方法,因为不能创建对象,所以设置为静态方法,通过类名获取这个数据库连接池实例static ConnectionPool *getConnectPool();// 删除拷贝构造函数和 "=" 操作符重载,因为要保证只有一个连接池实例ConnectionPool(const ConnectionPool &obj) = delete;ConnectionPool &operator=(const ConnectionPool &obj) = delete;// 外部调用这个,取出一个数据库连接shared_ptr<MysqlConn> getConnection();~ConnectionPool();private:// 构造函数私有化,不允许外部创建,因为保持一个对象,单例模式ConnectionPool();// 解析 json数据bool parseJsonFile();// 创建新的连接的线程处理动作函数void produceConnection();// 销毁连接线程的线程处理动作函数void recycleConnection();// 增加连接void addConnection();string m_ip;            // 数据发服务器的 IP地址string m_user;          // 数据库服务器用户名string m_passwd;        // 对应的密码string m_dbName;        // 数据库服务器上对应的数据库名unsigned short m_port;  // 数据库服务器端口号int m_minSize;          // 连接数量的最小值,这个是维护队列的最小值,保证突然的大量连接请求int m_maxSize;          // 连接数量的最大值,这个是队列数量 + m_busySize,表示最多只支持的连接数,包括了正在使用的atomic<int> m_busySize; // 正在忙的连接数,这个是共享资源,设置为原子变量int m_timeout;          // 超时时长,当连接数用完后等待的阻塞时长int m_maxIdleTime;      // 最大空闲时长,决定是否断开这个数据库连接// 连接池队列,存储的时是若干个数据库连接queue<MysqlConn *> m_connectionQ;mutex m_mutexQ;            // 连接池队列的互斥锁condition_variable m_cond; // 条件变量
};

构造函数

  • 首先解析 json 格式数据,设置端口号和 IP
  • 然后以最小连接数添加连接到队列
  • 创建两个线程进行管理添加连接和销毁连接的实现

添加连接

  • 这个使用一个线程单独维护,一直运行,除非连接池发送一个关闭信号
  • 持续判断能否添加连接
// 创建新的连接
void ConnectionPool::produceConnection()
{while (true){if (isShutdown)return;// 最小连接数,保证队列里有最小连接数,即使不使用,应对突然的高数量访问unique_lock<mutex> locker(m_mutexQ);while (m_connectionQ.size() >= m_minSize || m_connectionQ.size() + m_busySize >= m_maxSize){m_cond.wait(locker);// 这个非常重要,如果是因为连接池关闭唤醒,直接退出if (isShutdown)return;}addConnection();     // 将连接加入队列 (封装的函数)m_cond.notify_all(); // 唤醒阻塞在队列为空的线程}
}
  • 如果队列的数量比最小连接数大,就不需要添加连接
  • 如果队列的连接数量 + 忙的连接 (正在被使用的) 超过了连接上限,就不需要往队列里生产连接了,即使此时队列的数量比最小连接数小,仍不能添加连接,因为添加连接后,客户端就能取,如果都取了,就超过连接上限了,因此维护的是队列连接数 + 忙的连接数要小于最大连接上限
  • 也就是只有队列的数量比最小连接数小,并且此时队列的连接数量 + 忙的连接 (正在被使用的) 小于连接上限,才添加连接到队列

销毁连接

  • 也是使用一个线程隔一段时间就进行检测,销毁的连接是队列中的空闲连接
  • 因此取头部连接,计算它的存活时间,如果大于我们设定的值,就销毁这个连接,弹出队列,销毁它是因为它一直没被使用,属于空闲连接,只需一直销毁到保持有最小连接数即可
  • 注意销毁弹出之前,需要满足队列的数量大于最小连接数,如果小于,就不进行这个操作
// 销毁连接
// 销毁的空闲连接,满足超过一定时长还是空闲就销毁
void ConnectionPool::recycleConnection()
{while (true){if (isShutdown)return;this_thread::sleep_for(chrono::milliseconds(500));unique_lock<mutex> locker(m_mutexQ);while (m_connectionQ.size() > m_minSize){MysqlConn *conn = m_connectionQ.front();// 存活的时间if (conn->getAliveTime() >= m_maxIdleTime){m_connectionQ.pop();delete conn;}elsebreak; }}
}

取连接

  • 首先需要检测这个队列是否为空,和现在忙的 (即正在使用的连接) 数量是否大于了连接上限,如果满足这二者之中的任何一个,都对其进行阻塞等待,不取连接
  • 这个阻塞设置一个超时时间,如果满足上述二者只中一个,就阻塞,达到阻塞时长后,这个阻塞会解除,自己进行选择是退出还是继续进行阻塞
  • 取连接后,由于这个连接不使用需要回收,重新加入队列中,因此需要保存这个连接,这个是一个指针,因此考虑共享智能指针维护,更安全
  • 并且这个共享智能指针还有个功能,可以指定删除器函数,也就是这个指针生命周期结束后的指针处理,默认是清除智能指针指向的地址,由于这里不是进行销毁,我们需要重新加入队列,因此重定义删除器函数,使之为这个指针 (这个数据库连接)重新加入队列
// 外部调用这个,取出一个数据库连接
shared_ptr<MysqlConn> ConnectionPool::getConnection()
{// 这个能判断四种情况// 1. 如果没达到最大连接数请求,但是队列为空,阻塞// 2. 如果没达到最大连接数请求,但是队列不为空,不阻塞// 3. 如果达到最大连接数请求,但是队列不为空,阻塞// 4. 如果达到最大连接s数请求,但是队列为空,阻塞// 只要没达到最大连接数以及队列不为空,才能取连接,否则阻塞// 设置的超时检测,超过时长没被唤醒,则继续执行 while 判断是否继续阻塞unique_lock<mutex> locker(m_mutexQ);while (m_connectionQ.empty() || (m_busySize >= m_maxSize)){// 阻塞指定的时间长度,时间到了就解除阻塞if (cv_status::timeout == m_cond.wait_for(locker, chrono::milliseconds(m_timeout))){// 进入这个 if 说明阻塞 m_timeout 这个时长,还是没有被唤醒if (m_connectionQ.empty() || (m_busySize >= m_maxSize)){// 两种方式,continue继续阻塞,或者 return退出// return nullptr;continue;}}}// 使用共享的智能指针,回收当前连接用完后,重新 Push进队列中,自动回收// 由于这个共享智能指针默认删除器处理动作是:回收这个智能指针指向的内存地址// 但是我们不是要回收,我们是要重新加入队列,因此重新指定删除器函数// 也就是第二个参数,这里使用 lambda方式实现shared_ptr<MysqlConn> connptr(m_connectionQ.front(),[this](MysqlConn *conn){unique_lock<mutex> locker(m_mutexQ);// 更新加入队列的时间conn->refreshAliveTime();m_connectionQ.push(conn);m_busySize--;m_cond.notify_all(); // 唤醒因最大连接数阻塞的线程});m_connectionQ.pop();m_busySize++;// 本意是唤醒生产者线程,也就是唤醒创建新连接的线程// 虽然会唤醒取连接的线程,但是不影响,它仍会继续阻塞m_cond.notify_all();return connptr;
}

解析 Json

  • 为了不写死这个数据库连接的 IP 和端口号,这里使用 json 格式读取进去赋值
// 解析 json数据
bool ConnectionPool::parseJsonFile()
{ifstream ifs("dbconf.json");Reader rd;Value root;rd.parse(ifs, root);if (root.isObject()){m_ip = root["ip"].asString();m_port = root["port"].asInt();m_user = root["userName"].asString();m_passwd = root["password"].asString();m_dbName = root["dbName"].asString();m_minSize = root["minSize"].asInt();m_maxSize = root["maxSize"].asInt();m_maxIdleTime = root["maxIdleTime"].asInt();m_timeout = root["timeout"].asInt();m_busySize = 0;return true;}return false;
}

说明: 参考学习 https://www.bilibili.com/video/BV1Fr4y1s7w4/?spm_id_from=333.999.0.0

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

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

相关文章

接上文实现SpringSecurity,拦截器的实现

实现拦截器有图片可知,在上篇文章我们重写了UserDetailsManager,现在我们来进行之后的操作 在UserDetailsManager中我们可以调动数据库去进行一个账号密码的校验 之后我们这样设置拦截器进行一个token获取存储在usernamePasswordAuthenticationFilter这一层中, 有,则存储在Secu…

error: index-pack died of signal 15

使用Jenkins从gitlab拉取代码构建时,遇到报错error: index-pack died of signal 15一个常见的原因是仓库过大或网络不佳引起的超时,以下配置可解决。 配置工程,为源码管理部分增加Additional Behaviours,选择“高级的克隆行为”,把克隆和拉取操作的超时时间(分钟)设置为…

CentOS 7.9安装ElasticSearch7.14.0、ElasticSearch-Head、Kibana、Node14.18.2

CentOS 7.9安装ElasticSearch7.14.0、ElasticSearch-Head、Kibana、Node14.18.21.安装文件1. elasticsearch-7.14.0-linux-x86 64.tar.gz 2. elasticsearch-head-master.zip 3. jdk-11 linux-x64 bin.tar.gz 4. kibana-7.14.0-linux-x86 64.tar.gz 5. node-v14.18.2-linux-x64.…

闲话 9.29(更新了鲜花部分)

杂项乱写 9.29因为没有模拟赛,所以考虑捡捡之前漏下的小点。 // Upd:这样的标题看的人应该会多些? 更新了鲜花 注:LCA 之后的讲解中可能会出现一些自由的文字,酌情阅读。Better DaysRun away, run away, yeah Take your time to say a prayer Recognize me when you sneak…

南沙C++信奥赛陈老师解一本通题 1973:【16NOIP普及组】买铅笔

​【题目描述】P老师需要去商店买n支铅笔作为小朋友们参加NOIP的礼物。她发现商店一共有3种包装的铅笔,不同包装内的铅笔数量有可能不同,价格也有可能不同。为了公平起见,P老师决定只买同一种包装的铅笔。 商店不允许将铅笔的包装拆开,因此P老师可能需要购买超过n支铅笔才够…

城市空间设计对居民生活质量的影响:构建宜居城市的蓝图

在快节奏的现代生活中,城市不仅是经济活动的中心,更是人们生活、工作、休闲的综合载体。本文旨在深入探讨城市空间设计如何通过科学规划、人性化考量以及生态融合,为居民打造更加宜居、和谐的生活环境。 1. 促进社区互动与归属感城市空间设计首先关注的是人与人之间的联系。…

「土地那些事儿」我国土地资源的特点:多样而丰富

土地,作为自然资源的核心,承载着国家的经济发展、社会进步和生态安全。我国作为世界上人口最多、面积第三大的国家,土地资源的特点尤为显著。站在这片古老而又充满活力的土地上,我们不禁会思考:我国的土地资源有哪些独特之处?今天,就让我们一起走进这片广袤的土地,探寻…

三维立体自然资源“一张图”

随着信息技术的发展,自然资源管理迎来了新的机遇与挑战。在众多技术中,“三维立体自然资源‘一张图’”的概念尤为引人注目。它不仅代表了地理信息科学领域的最新成果,也为自然资源的有效管理和可持续利用提供了强有力的支持。本文将探讨这一概念的内涵及其在自然资源管理中…

从Anaconda到PyTorch到训练Yolo——Windows系统

1、Anaconda conda能管理不同的开发编译环境,互补干涉影响。 Anaconda和Miniconda是conda的工具,前者带界面(大而全),后者只有命令行窗口(小而精)。作为初学,建议安装Anaconda 1.1 安装Anaconda 下载 Download Anaconda Distribution | Anaconda ,安装到D盘,其他默认…

9月28日,工信部人才交流中心CUUG-PGCA/PGCP/PGCM认证考试完成!

2024年9月28日,由工业和信息化部人才交流中心主办,北京神脑资讯技术有限公司承办的PostgreSQL管理员岗位能力认证考试(PGCP中级/PGCM高级)顺利完成。 中级PG认证专家:PGCP(PostgreSQL Certified Professional),是对PostgreSQL数据库技术能力的一种认可,达到了专家级别…

设置GRUB密码

通过在GRUB配置中设置密码,系统在启动时会在显示GRUB菜单之前提示用户输入密码,只有输入正确的密码后,用户才能看到并修改启动选项。 1、以 root 用户身份执行 grub2-setpassword 命令。2、检查 /boot/grub2/user.cfg 文件,其包含哈希格式的密码。3、配置GRUB菜单作者:杨灏…

模糊查询用逗号分隔开的字段

业务:一个sys_dept部门表中,有一个字段ancestors是用逗号分隔开的id。 想模糊查询这个字段。 如果用普通的like的话。 select * from sys_dept where ancestors like %1%实现不了我们想要的效果。 mysql 可以用FIND_IN_SET()这个函数来协助。 列:SELECT * FROM sys_dept …