前言
数据库连接池的基本思想是:为数据库连接建立一个“缓冲池”,预先在池中放入一定数量的数据库连接管道,需要时,从池子中取出管道进行使用,操作完毕后,再将管道放入池子中,从而避免了频繁的向数据库申请资源,释放资源带来的性能损耗。在如今的分布式系统当中,系统的QPS瓶颈往往就在数据库,所以理解数据库连接池底层构造原理与设计思想是很有益处的。我们常用的数据库连接池有C3P0,DBCP,Druid等,下面我们就来分析下数据库连接池应该有些什么,以及手写一个迷你版的数据库连接
对数据库连接池的一点思考
![5766fd60bb84145092007b6200e5d7c1.png](https://i-blog.csdnimg.cn/blog_migrate/750f3aa462a68a0fc23f4085b5e559e4.jpeg)
数据库连接池
第一,数据库连接池中存放的就是数据库操作管道,不仅仅是存放,而且应该是管理这些管道;
第二,应该提供外部配置文件去初始化数据库连接池;
第三,如果一个数据库操作管道已经被占用,那么其他请求是否应该得到这个管道,也就是说我们要考虑多线程并发下,管道的分配问题;
第四,如果做到管道的复用?放回池子中,表示可用,并不是真正的关闭管道;
写一个迷你版数据库连接池
![4dd1889354e4886f9bd775acfe06376f.png](https://i-blog.csdnimg.cn/blog_migrate/67141d5577c06b0057af1df09771aa17.jpeg)
工程类关系图说明
IMyPool是一个接口,对外提供数据库连接池的基本服务,比如得到一个数据库操作管道。
MyDefaultPool是IMyPool的实现。
MyPooledConnection代表数据库操作管道,它可以执行SQL,关闭管道等。
MyPoolFactory是一个工厂,单例模式,用于得到IMyPool实现。
DBConfigXML代表外部配置文件。
Test用于测试。
DBConfigXML
![66d4e0b74cac49d62684d18a07bae084.png](https://i-blog.csdnimg.cn/blog_migrate/9456ecca37f74d3974b8cb274407994e.jpeg)
DBConfigXML
我们在实际中使用数据库连接池,需要在Spring的配置文件中,进行一些参数配置。这里,为了简化解析,直接提供。
MyPooledConnection
![b33eb72505faf6899d68338583cdd3b4.png](https://i-blog.csdnimg.cn/blog_migrate/0585028e76c247fd39f53edc7f50b0ca.jpeg)
数据库连接管道
所谓数据库连接管道,就是对JDBC Connection进行封装而已,但是需要注意isBusy的这个标示。对管道的关闭,实际上只是标示的改变而已!
IMyPool
![e0c6ddd1931d9e635c49c84ef4cc34d9.png](https://i-blog.csdnimg.cn/blog_migrate/2983f3482ba802602460777e5a3d8874.jpeg)
IMyPool
MyDefaultPool
![7f63e11ddd8f72cd755dc23d3fbeee70.png](https://i-blog.csdnimg.cn/blog_migrate/21658d6d4b3b7e9d05d5ffd96e36b26f.jpeg)
MyDefaultPool
需要注意到是,MyDefaultPool持有一个管道集合,基于多线程的考虑,这里使用了Vector。
MyDefaultPool需要初始化
![c6231652d701a5471a856c37321534c2.png](https://i-blog.csdnimg.cn/blog_migrate/a163b8a9ff079d342d6685b0ac930868.jpeg)
数据库连接池初始化
数据库连接池需要根据外部配置文件完成数据库驱动加载以及初始化管道的建立。
createMyPooledConnection接口实现
![5de24c827d364e51ef665d835b639e6d.png](https://i-blog.csdnimg.cn/blog_migrate/48925742eb8c15c74340674874d1634a.jpeg)
创建数据库连接管道
数据库连接池在创建管道时,应该去看一下是否达到上限,如果没有,则可以创建。
不仅仅要创建出来,还要标示每一个管道的isBusy标志。
getMyPooledConnection接口实现
![3163d355e7f9b2ffb3c50c328af6c2f2.png](https://i-blog.csdnimg.cn/blog_migrate/e4119aeacf79b258c8f7b062e13f4a45.jpeg)
得到数据库连接管道
这里需要注意的是:如果得不到操作管道,需要去创建管道!
getRealConnectionFromPool
![5dc4a80fe2a32109c847a3c748d18aeb.png](https://i-blog.csdnimg.cn/blog_migrate/01b4a135015d9f1ff32fbd2e983259ea.jpeg)
getRealConnectionFromPool
第一,这里使用了synchronized,就是为了避免多线程下产生问题。
第二,要知道Connection是有超时机制的,如果我们得到的管道的Connection已经超时了怎么办呢?
第三,得到管道后,一定注意isBusy的设置。
MyPoolFactory
![7e2c84ef83569b6ffe8c343fea95dada.png](https://i-blog.csdnimg.cn/blog_migrate/b0612c34fae7ef509f776a7ac322a3f5.jpeg)
单例工厂
Test测试
![043c8e20cd2f6abe3c8469be54c71248.png](https://i-blog.csdnimg.cn/blog_migrate/abc7b1841d501ecac03aa7855e520d6c.jpeg)
测试类
运行结果
![ed11998451a4387660f7c3c913d02eac.png](https://i-blog.csdnimg.cn/blog_migrate/e0d31962a4460b4e5e49f078b64b8480.jpeg)
好了,到这里,一个迷你版的数据库连接池就有模有样了!