自定义数据库连接池

数据库连接池负责分配、管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个,应用程序和数据库创建连接需要消耗很大的资源。

package com.lbx.myDataSource;

import javax.sql.DataSource;
import java.io.PrintWriter;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.util.LinkedList;
import java.util.logging.Logger;

/**
 * @Author:
 * @date 2018/4/2 9:53
 */
public class MyDataSource implements DataSource {
    //数据库连接信息
    String url = "jdbc:mysql://172.31.12.180:3306/bos";
    String user = "root";
    String password = "liaoBAOxin123...";
    //创建一个连接池,因为频繁需要增删,所以选择链表数据结构,LinkedList
    LinkedList<Connection> poll = new LinkedList<Connection>();

    public MyDataSource() {
        try {
            //注册驱动
            Class.forName("com.mysql.jdbc.Driver");
            //初始化两条连接
            for (int i = 0; i < 2; i++) {
                //创建连接
                Connection conn = DriverManager.getConnection(url, user, password);
                poll.add(conn);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Override
    public Connection getConnection() throws SQLException {
        //并发情况下,只能一个线程进去
        synchronized (this) {
            //如果池中没有连接,则创建连接
            if (poll.isEmpty()) {
                for (int i = 0; i < 2; i++) {
                    //创建连接
                    Connection conn = DriverManager.getConnection(url, user, password);
                    poll.add(conn);
                }
            }
            //removeFirst执行方法的返回值是connection
            final Connection conn = poll.removeFirst();
            //利用动态代理技术,增强close方法
            Connection proxyconn = (Connection) Proxy.newProxyInstance(conn.getClass().getClassLoader(), conn.getClass().getInterfaces(), new InvocationHandler() {
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                    if (method.getName().equals("close")) {
                        System.out.println("方法名称" + method.getName());
                        System.out.println("连接给回给连接池");
                        return poll.add(conn);
                    } else {
                        return method.invoke(conn, args);
                    }
                }
            });
            return proxyconn;
        }
    }

    @Override
    public Connection getConnection(String username, String password) throws SQLException {
        return null;
    }

    @Override
    public <T> T unwrap(Class<T> iface) throws SQLException {
        return null;
    }

    @Override
    public boolean isWrapperFor(Class<?> iface) throws SQLException {
        return false;
    }

    @Override
    public PrintWriter getLogWriter() throws SQLException {
        return null;
    }

    @Override
    public void setLogWriter(PrintWriter out) throws SQLException {

    }

    @Override
    public void setLoginTimeout(int seconds) throws SQLException {

    }

    @Override
    public int getLoginTimeout() throws SQLException {
        return 0;
    }

    @Override
    public Logger getParentLogger() throws SQLFeatureNotSupportedException {
        return null;
    }
}

“`

package com.lbx.myDataSource; import org.junit.Test; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; public class MyTest {

@Test
public void fun1() throws SQLException {
    Connection conn = null;
    PreparedStatement pstmt = null;
    // 1.创建自定义连接池对象
    MyDataSource dataSource = new MyDataSource();
    try {
        // 2.从池子中获取连接
        conn = dataSource.getConnection();
        String sql = "insert into t_user values(?)";
        pstmt = conn.prepareStatement(sql);
        pstmt.setString(1, "张三");
        int rows = pstmt.executeUpdate();
        if (rows > 0) {
            System.out.println("添加成功!");
        } else {
            System.out.println("添加失败!");
        }
    } catch (Exception e) {
        throw new RuntimeException(e);
    }finally {
        conn.close();
    }

}

}

源码 https://github.com/liaobaoxin/MyDataSource

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏芋道源码1024

数据库中间件 Sharding-JDBC 源码分析 —— JDBC实现与读写分离

本文主要基于 Sharding-JDBC 1.5.0 正式版 1. 概述 2. unspported 包 3. adapter 包 3.1 WrapperAda...

38880
来自专栏Android干货

Java数据库编程--JDBC

492120
来自专栏与神兽党一起成长

jFinal路由解析源码分析

jFinal的路由解析是在JFinalFilter中做的,这个Filter也需要在web.xml中配置。JFinalFilter实现了javax.servlet...

17020
来自专栏Hongten

单独java 程序连接Mysql数据库

2.下载mysql驱动的jar包,我的版本为:mysql-connector-java-5.1.10-bin.jar

15830
来自专栏JackieZheng

开发人员看测试之细说JBehave

上篇我们说到如何从Github上clone出一个JBehave项目,既是为了学习JBehava,也是为了熟悉下Github。 从clone下来的项目看来,基本没...

313100
来自专栏技术碎碎念

jdbc基础 (二) 通过properties配置文件连接数据库

上一篇描述了对mysql数据库的简单操作,下面来看一下开发中应该如何灵活应用。 因为jdbc对数据库的驱动加载、连接获取、释放资源的代码都是相同的,为了提高代码...

30780
来自专栏Java后端生活

JDBC(十)DBUtils

commons-dbutils 是 Apache 组织提供的一个开源 JDBC工具类库,它是对JDBC的简单封装,学习成本极低,并且使用dbutils能极大简化...

15030
来自专栏逆向技术

内核开发知识第一讲.内核中的数据类型.重要数据结构.常用内核API函数.

  在内核中.程序的编写不能简单的用基本数据类型了. 因为操作系统不同.很有可能造成数据类型的长度不一.而产生重大问题.所以在内核中.

19320
来自专栏史上最简单的Spring Cloud教程

如何用Redlock实现分布式锁

之前写过一篇文章《如何在springcloud分布式系统中实现分布式锁?》,由于自己仅仅是阅读了相关的书籍,和查阅了相关的资料,就认为那样的是可行的。那篇文章实...

56170
来自专栏Java帮帮-微信公众号-技术文章全总结

JavaWeb07-JDBC(Java真正的全栈开发)

? jdbc 一、JDBC介绍 1. JDBC定义 JDBC(Java Data Base Connectivity,java数据库连接),说白了就是用Jav...

40260

扫码关注云+社区

领取腾讯云代金券