JAVA数据库连接池实现
转载自 JAVA数据库连接池实现
连接池的管理用了了享元模式,这里对连接池进行简单设计。
一、设计思路
1.连接池配置属性DBbean:里面存放可以配置的一些属性
2.连接池接口IConnectionPool:里面定义一些基本的获取连接的一些方法
3.接口实现ConnectionPool:对上面操作进行实现,并加入一些其他方法
4.连接池管理ConnectionPoolManager:管理所有的不同的连接池,所有的连接都能通过这里进行获得连接
5.另外还有几个测试类,和连接信息模拟的类,这里就不进行xml 和配置文件信息的读取了
- package pool;
- /**
- * 这是外部可以配置的连接池属性
- * 可以允许外部配置,拥有默认值
- * @author Ran
- *
- */
- public class DBbean {
- // 连接池属性
- private String driverName;
- private String url;
- private String userName;
- private String password;
- // 连接池名字
- private String poolName;
- private int minConnections = 1; // 空闲池,最小连接数
- private int maxConnections = 10; // 空闲池,最大连接数
- private int initConnections = 5;// 初始化连接数
- private long connTimeOut = 1000;// 重复获得连接的频率
- private int maxActiveConnections = 100;// 最大允许的连接数,和数据库对应
- private long connectionTimeOut = 1000*60*20;// 连接超时时间,默认20分钟
- private boolean isCurrentConnection = true; // 是否获得当前连接,默认true
- private boolean isCheakPool = true; // 是否定时检查连接池
- private long lazyCheck = 1000*60*60;// 延迟多少时间后开始 检查
- private long periodCheck = 1000*60*60;// 检查频率
- public DBbean(String driverName, String url, String userName,
- String password, String poolName) {
- super();
- this.driverName = driverName;
- this.url = url;
- this.userName = userName;
- this.password = password;
- this.poolName = poolName;
- }
- public DBbean() {
- }
- public String getDriverName() {
- if(driverName == null){
- driverName = this.getDriverName()+"_"+this.getUrl();
- }
- return driverName;
- }
- public void setDriverName(String driverName) {
- this.driverName = driverName;
- }
- public String getUrl() {
- return url;
- }
- public void setUrl(String url) {
- this.url = url;
- }
- public String getUserName() {
- return userName;
- }
- public void setUserName(String userName) {
- this.userName = userName;
- }
- public String getPassword() {
- return password;
- }
- public void setPassword(String password) {
- this.password = password;
- }
- public String getPoolName() {
- return poolName;
- }
- public void setPoolName(String poolName) {
- this.poolName = poolName;
- }
- public int getMinConnections() {
- return minConnections;
- }
- public void setMinConnections(int minConnections) {
- this.minConnections = minConnections;
- }
- public int getMaxConnections() {
- return maxConnections;
- }
- public void setMaxConnections(int maxConnections) {
- this.maxConnections = maxConnections;
- }
- public int getInitConnections() {
- return initConnections;
- }
- public void setInitConnections(int initConnections) {
- this.initConnections = initConnections;
- }
- public int getMaxActiveConnections() {
- return maxActiveConnections;
- }
- public void setMaxActiveConnections(int maxActiveConnections) {
- this.maxActiveConnections = maxActiveConnections;
- }
- public long getConnTimeOut() {
- return connTimeOut;
- }
- public void setConnTimeOut(long connTimeOut) {
- this.connTimeOut = connTimeOut;
- }
- public long getConnectionTimeOut() {
- return connectionTimeOut;
- }
- public void setConnectionTimeOut(long connectionTimeOut) {
- this.connectionTimeOut = connectionTimeOut;
- }
- public boolean isCurrentConnection() {
- return isCurrentConnection;
- }
- public void setCurrentConnection(boolean isCurrentConnection) {
- this.isCurrentConnection = isCurrentConnection;
- }
- public long getLazyCheck() {
- return lazyCheck;
- }
- public void setLazyCheck(long lazyCheck) {
- this.lazyCheck = lazyCheck;
- }
- public long getPeriodCheck() {
- return periodCheck;
- }
- public void setPeriodCheck(long periodCheck) {
- this.periodCheck = periodCheck;
- }
- public boolean isCheakPool() {
- return isCheakPool;
- }
- public void setCheakPool(boolean isCheakPool) {
- this.isCheakPool = isCheakPool;
- }
- }
- package pool;
- import java.sql.Connection;
- import java.sql.SQLException;
- public interface IConnectionPool {
- // 获得连接
- public Connection getConnection();
- // 获得当前连接
- public Connection getCurrentConnecton();
- // 回收连接
- public void releaseConn(Connection conn) throws SQLException;
- // 销毁清空
- public void destroy();
- // 连接池是活动状态
- public boolean isActive();
- // 定时器,检查连接池
- public void cheackPool();
- }
- package pool;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.SQLException;
- import java.util.List;
- import java.util.Timer;
- import java.util.TimerTask;
- import java.util.Vector;
- public class ConnectionPool implements IConnectionPool {
- // 连接池配置属性
- private DBbean dbBean;
- private boolean isActive = false; // 连接池活动状态
- private int contActive = 0;// 记录创建的总的连接数
- // 空闲连接
- private List<Connection> freeConnection = new Vector<Connection>();
- // 活动连接
- private List<Connection> activeConnection = new Vector<Connection>();
- // 将线程和连接绑定,保证事务能统一执行
- private static ThreadLocal<Connection> threadLocal = new ThreadLocal<Connection>();
- public ConnectionPool(DBbean dbBean) {
- super();
- this.dbBean = dbBean;
- init();
- cheackPool();
- }
- // 初始化
- public void init() {
- try {
- Class.forName(dbBean.getDriverName());
- for (int i = 0; i < dbBean.getInitConnections(); i++) {
- Connection conn;
- conn = newConnection();
- // 初始化最小连接数
- if (conn != null) {
- freeConnection.add(conn);
- contActive++;
- }
- }
- isActive = true;
- } catch (ClassNotFoundException e) {
- e.printStackTrace();
- } catch (SQLException e) {
- e.printStackTrace();
- }
- }
- // 获得当前连接
- public Connection getCurrentConnecton(){
- // 默认线程里面取
- Connection conn = threadLocal.get();
- if(!isValid(conn)){
- conn = getConnection();
- }
- return conn;
- }
- // 获得连接
- public synchronized Connection getConnection() {
- Connection conn = null;
- try {
- // 判断是否超过最大连接数限制
- if(contActive < this.dbBean.getMaxActiveConnections()){
- if (freeConnection.size() > 0) {
- conn = freeConnection.get(0);
- if (conn != null) {
- threadLocal.set(conn);
- }
- freeConnection.remove(0);
- } else {
- conn = newConnection();
- }
- }else{
- // 继续获得连接,直到从新获得连接
- wait(this.dbBean.getConnTimeOut());
- conn = getConnection();
- }
- if (isValid(conn)) {
- activeConnection.add(conn);
- contActive ++;
- }
- } catch (SQLException e) {
- e.printStackTrace();
- } catch (ClassNotFoundException e) {
- e.printStackTrace();
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- return conn;
- }
- // 获得新连接
- private synchronized Connection newConnection()
- throws ClassNotFoundException, SQLException {
- Connection conn = null;
- if (dbBean != null) {
- Class.forName(dbBean.getDriverName());
- conn = DriverManager.getConnection(dbBean.getUrl(),
- dbBean.getUserName(), dbBean.getPassword());
- }
- return conn;
- }
- // 释放连接
- public synchronized void releaseConn(Connection conn) throws SQLException {
- if (isValid(conn)&& !(freeConnection.size() > dbBean.getMaxConnections())) {
- freeConnection.add(conn);
- activeConnection.remove(conn);
- contActive --;
- threadLocal.remove();
- // 唤醒所有正待等待的线程,去抢连接
- notifyAll();
- }
- }
- // 判断连接是否可用
- private boolean isValid(Connection conn) {
- try {
- if (conn == null || conn.isClosed()) {
- return false;
- }
- } catch (SQLException e) {
- e.printStackTrace();
- }
- return true;
- }
- // 销毁连接池
- public synchronized void destroy() {
- for (Connection conn : freeConnection) {
- try {
- if (isValid(conn)) {
- conn.close();
- }
- } catch (SQLException e) {
- e.printStackTrace();
- }
- }
- for (Connection conn : activeConnection) {
- try {
- if (isValid(conn)) {
- conn.close();
- }
- } catch (SQLException e) {
- e.printStackTrace();
- }
- }
- isActive = false;
- contActive = 0;
- }
- // 连接池状态
- @Override
- public boolean isActive() {
- return isActive;
- }
- // 定时检查连接池情况
- @Override
- public void cheackPool() {
- if(dbBean.isCheakPool()){
- new Timer().schedule(new TimerTask() {
- @Override
- public void run() {
- // 1.对线程里面的连接状态
- // 2.连接池最小 最大连接数
- // 3.其他状态进行检查,因为这里还需要写几个线程管理的类,暂时就不添加了
- System.out.println("空线池连接数:"+freeConnection.size());
- System.out.println("活动连接数::"+activeConnection.size());
- System.out.println("总的连接数:"+contActive);
- }
- },dbBean.getLazyCheck(),dbBean.getPeriodCheck());
- }
- }
- }
- package pool;
- import java.sql.Connection;
- import java.sql.SQLException;
- import java.util.Hashtable;
- /**
- * 连接管理类
- * @author Ran
- *
- */
- public class ConnectionPoolManager {
- // 连接池存放
- public Hashtable<String,IConnectionPool> pools = new Hashtable<String, IConnectionPool>();
- // 初始化
- private ConnectionPoolManager(){
- init();
- }
- // 单例实现
- public static ConnectionPoolManager getInstance(){
- return Singtonle.instance;
- }
- private static class Singtonle {
- private static ConnectionPoolManager instance = new ConnectionPoolManager();
- }
- // 初始化所有的连接池
- public void init(){
- for(int i =0;i<DBInitInfo.beans.size();i++){
- DBbean bean = DBInitInfo.beans.get(i);
- ConnectionPool pool = new ConnectionPool(bean);
- if(pool != null){
- pools.put(bean.getPoolName(), pool);
- System.out.println("Info:Init connection successed ->" +bean.getPoolName());
- }
- }
- }
- // 获得连接,根据连接池名字 获得连接
- public Connection getConnection(String poolName){
- Connection conn = null;
- if(pools.size()>0 && pools.containsKey(poolName)){
- conn = getPool(poolName).getConnection();
- }else{
- System.out.println("Error:Can't find this connecion pool ->"+poolName);
- }
- return conn;
- }
- // 关闭,回收连接
- public void close(String poolName,Connection conn){
- IConnectionPool pool = getPool(poolName);
- try {
- if(pool != null){
- pool.releaseConn(conn);
- }
- } catch (SQLException e) {
- System.out.println("连接池已经销毁");
- e.printStackTrace();
- }
- }
- // 清空连接池
- public void destroy(String poolName){
- IConnectionPool pool = getPool(poolName);
- if(pool != null){
- pool.destroy();
- }
- }
- // 获得连接池
- public IConnectionPool getPool(String poolName){
- IConnectionPool pool = null;
- if(pools.size() > 0){
- pool = pools.get(poolName);
- }
- return pool;
- }
- }
- package pool;
- import java.util.ArrayList;
- import java.util.List;
- /**
- * 初始化,模拟加载所有的配置文件
- * @author Ran
- *
- */
- public class DBInitInfo {
- public static List<DBbean> beans = null;
- static{
- beans = new ArrayList<DBbean>();
- // 这里数据 可以从xml 等配置文件进行获取
- // 为了测试,这里我直接写死
- DBbean beanOracle = new DBbean();
- beanOracle.setDriverName("oracle.jdbc.driver.OracleDriver");
- beanOracle.setUrl("jdbc:oracle:thin:@7MEXGLUY95W1Y56:1521:orcl");
- beanOracle.setUserName("mmsoa");
- beanOracle.setPassword("password1234");
- beanOracle.setMinConnections(5);
- beanOracle.setMaxConnections(100);
- beanOracle.setPoolName("testPool");
- beans.add(beanOracle);
- }
- }
测试:
- package pool;
- import java.sql.Connection;
- /**
- * 模拟线程启动,去获得连接
- * @author Ran
- *
- */
- public class ThreadConnection implements Runnable{
- private IConnectionPool pool;
- @Override
- public void run() {
- pool = ConnectionPoolManager.getInstance().getPool("testPool");
- }
- public Connection getConnection(){
- Connection conn = null;
- if(pool != null && pool.isActive()){
- conn = pool.getConnection();
- }
- return conn;
- }
- public Connection getCurrentConnection(){
- Connection conn = null;
- if(pool != null && pool.isActive()){
- conn = pool.getCurrentConnecton();
- }
- return conn;
- }
- }
- package pool;
- public class Client {
- public static void main(String[] args) throws InterruptedException {
- // 初始化连接池
- Thread t = init();
- t.start();
- t.join();
- ThreadConnection a = new ThreadConnection();
- ThreadConnection b = new ThreadConnection();
- ThreadConnection c = new ThreadConnection();
- Thread t1 = new Thread(a);
- Thread t2 = new Thread(b);
- Thread t3 = new Thread(c);
- // 设置优先级,先让初始化执行,模拟 线程池 先启动
- // 这里仅仅表面控制了,因为即使t 线程先启动,也不能保证pool 初始化完成,为了简单模拟,这里先这样写了
- t1.setPriority(10);
- t2.setPriority(10);
- t3.setPriority(10);
- t1.start();
- t2.start();
- t3.start();
- System.out.println("线程A-> "+a.getConnection());
- System.out.println("线程B-> "+b.getConnection());
- System.out.println("线程C-> "+c.getConnection());
- }
- // 初始化
- public static Thread init() {
- Thread t = new Thread(new Runnable() {
- @Override
- public void run() {
- IConnectionPool pool = initPool();
- while(pool == null || !pool.isActive()){
- pool = initPool();
- }
- }
- });
- return t;
- }
- public static IConnectionPool initPool(){
- return ConnectionPoolManager.getInstance().getPool("testPool");
- }
- }
小结 :
1.连接池诞生原因是,如果每次都从数据库获得连接,时间比较长,因此我们提前做建立一些连接,放在连接池里面,每次都从里面取
2.上面仅仅写了连接池基本原理,关于多线程下连接池的管理没写,后面对多线程操作熟练了添加吧
JAVA数据库连接池实现相关推荐
- 主流Java数据库连接池比较及前瞻
本文转载自微信公众号「工匠小猪猪的技术世界」 主流数据库连接池 常用的主流开源数据库连接池有C3P0.DBCP.Tomcat Jdbc Pool.BoneCP.Druid等 C3p0: 开源的JDBC ...
- Java数据库连接池实现原理
https://blog.csdn.net/tuke_tuke/article/details/51532510 一般来说,Java应用程序访问数据库的过程是: ①装载数据库驱动程序: ②通过jdbc ...
- 主流Java数据库连接池分析(C3P0,DBCP,TomcatPool,BoneCP,Druid)
http://developer.51cto.com/art/201807/579402.htm 主流数据库连接池 常用的主流开源数据库连接池有C3P0.DBCP.Tomcat Jdbc Pool.B ...
- Java数据库连接池--DBCP浅析
转载自 Java数据库连接池--DBCP浅析 前言对于数据库连接池, 想必大家都已经不再陌生, 这里仅仅设计Java中的两个常用数据库连接池: DBCP和C3P0(后续会更新). 一. 为何要使用 ...
- Java 数据库连接池的技术选型都应考虑哪些要素
为什么80%的码农都做不了架构师?>>> 数据库连接池是一个牵涉面很广的话题,对于大型系统,数据库连接池的好坏,关系到系统的性能和稳定性,因此,选好数据库连接池,是系统在架构时 ...
- mysql连接池源码_一个JAVA数据库连接池实现源码
原文链接:http://www.open-open.com/lib/view/open1410875608164.html // // 一个效果非常不错的JAVA数据库连接池. // from:htt ...
- java数据库连接池选择及开发配置
一.数据库连接池概述 数据库连接的建立是一种耗时.性能低.代价高的操作,频繁的数据库连接的建立和关闭极大的影响了系统的性能.数据库连接池是系统初始化过程中创建一定数量的数据库连接放于连接池中,当程序需 ...
- java 连接池 druid_从零开始学 Java - 数据库连接池的选择 Druid
我先说说数据库连接 数据库大家都不陌生,从名字就能看出来它是「存放数据的仓库」,那我们怎么去「仓库」取东西呢?当然需要钥匙啦!这就是我们的数据库用户名.密码了,然后我们就可以打开门去任意的存取东西了. ...
- java 数据库连接池 实例_java数据库连接池和数据库连接示例
import java.sql.Connection; import java.sql.SQLException; import javax.sql.DataSource; import com.mc ...
最新文章
- 2021年大数据Hadoop(十):HDFS的数据读写流程
- ASP.NET MVC Module
- isinfinite_Java Double类isInfinite()方法与示例
- 35. 复杂链表的复制
- android安装apk提示版本号不同,android 安装apk 遇到的问题
- android 速度传感器,Android实战技巧之四十二:加速度传感器
- 定时重启_SpringBoot基于数据库的定时任务实现方法
- 项目管理软件之禅道和JIRA的共同点与区别
- 微信小程序申请微信支付0.2费率商户号微信小程序接入开通流程
- Pyrene-PEG-Acid,芘丁酸聚乙二醇羧基,Pyrene-PEG-COOH
- 被指开除高级研究员,谷歌大神Jeff Dean回应:是她说不答应条件就离职
- OpenGL课程设计 三维图形交互程序 bunny兔+飞机模型
- 周记——20150907
- 扔掉代码,程序员月薪达到了10k+
- 沈剑架构师之路的分享-总结
- blog群发王(价值1980元)源代码提供 1
- php打印10以内减法表,10以内加减法口诀表(A4纸可以打印)
- 手机上的磁性传感技术
- web前端开发之vue基础
- Android_AsyncTaskDemo之QQ记步数(画圆形图片知识)
热门文章
- 二叉树最近公共祖先相关题目(Leetcode题解-Python语言)
- 使用React hooks,些许又多了不少摸鱼时间
- 《C++ Primer》2.6.1节练习
- 中科大软件测试期末复习
- php mysql 子查询_php – MySQL查询和子查询
- cubemx lan8720模块_通过STM32cubeMX将STM32F767+LAN8720+LwIP+FreeRTOS的以太网实现
- PTA 三足鼎立 (lower_bound()+upper_bound())
- Spring boot——起步依赖
- 51 NOD 1363 最小公倍数之和 (欧拉函数思维应用)
- Harbour.Space Scholarship Contest 2021-2022 F. Pairwise Modulo 逆向思维 + 树状数组