MySQL学习(六)——自定义连接池

时间:2022-09-19 23:44:19

1、连接池概念

用池来管理Connection,这样可以重复使用Connection。有了池,我们就不用自己来创建Connection,而是通过池来获取Connection对象。当使用完Connection后,调用Connection的close()方法也不会真的关闭Connection,而是把Connection“归还”给池,池就可以再利用这个Connection对象了。

MySQL学习(六)——自定义连接池

2、自定义连接池

1)基于MySQL学习(五)——使用JDBC完成用户表CRUD的操作,把db.properties和JDBCUtils_V3.java文件复制到当前包下

2)MyConnection.java文件

装饰者设计模式,专门用于增强方法(此处如果不对close()做增强方法,用户调用conn.close()将连接真正释放,连接池将无连接可用,所以我们希望用户调用了close()方法,连接仍归还给连接池)

 1 //1.实现同一个接口
2 public class MyConnection implements Connection{
3 //3.定义变量
4 private Connection conn;
5
6 private LinkedList<Connection> pool;
7
8 //2.编写一个构造方法(参数使用了面向对象的多态特性)
9 public MyConnection(Connection conn,LinkedList<Connection> pool){
10 this.conn=conn;
11 this.pool=pool;
12 }
13 //4.书写需要增强的方法
14 @Override
15 public void close() throws SQLException {
16 pool.add(conn);
17 }
18 /*
19 * 此方法必须覆盖!否则会出现空指针异常
20 */
21 @Override
22 public PreparedStatement prepareStatement(String sql) throws SQLException {
23 return conn.prepareStatement(sql);
24 }
25 ……
26 }

3)MyDataSource1.java文件

创建连接池实现数据源,并实现接口javax.sql.DataSource

 1 public class MyDataSource1 implements DataSource{
2 //1.创建一个容器用于存储Connection对象
3 private static LinkedList<Connection> pool=new LinkedList<Connection>();
4 //2.创建5个连接放到容器中去
5 static{
6 for(int i=0;i<5;i++){
7 Connection conn=JDBCUtils_V3.getConnection();
8 //放入池子中的connection对象已经经过改造了
9 MyConnection myconn=new MyConnection(conn,pool);
10 pool.add(myconn);
11 }
12 }
13 /**
14 * 重写获取连接的方法
15 */
16 @Override
17 public Connection getConnection() throws SQLException {
18 Connection conn=null;
19 //3.使用前先判断
20 if(pool.size()==0){
21 //3.1池子里没有,我们再创建一些
22 conn=JDBCUtils_V3.getConnection();
23 pool.add(conn);
24 }
25 //4.从池子里面获取一个连接对象Connection
26 conn=pool.remove(0);
27 return conn;
28 }
29 ……
30 }

4)TestMyDataSource.java文件

 1 public class TestMyDataSource {
2 /*
3 * 添加用户
4 * 使用改造过后的Connection
5 */
6 @Test
7 public void testAddUser1(){
8 Connection conn=null;
9 PreparedStatement pstmt=null;
10 //1.创建自定义连接池对象
11 DataSource dataSource=new MyDataSource1();
12
13 try {
14 //2.从池子中获取连接
15 conn=dataSource.getConnection();
16 String sql="insert tbl_user values(null,?,?)";
17 //必须在自定义的conntection类中重写prepareStatement(sql)语句
18 pstmt=conn.prepareStatement(sql);
19 pstmt.setString(1, "chenga");
20 pstmt.setString(2, "123");
21 int rows=pstmt.executeUpdate();
22 if(rows>0){
23 System.out.println("添加成功");
24 }else{
25 System.out.println("添加不成功");
26 }
27 } catch (SQLException e) {
28 e.printStackTrace();
29 }finally{
30 //调用的release()方法中的close()其实是MyConnection的close()方法,该方法内部将当前连接归还到连接池
31 JDBCUtils_V3.release(conn,pstmt,null);
32 }
33 }
34 }

执行Junit Test,效果如下:

MySQL学习(六)——自定义连接池