J2EE开发技术点5:Tomcat jdbc pool

简介:

前言
数据库连接是一种昂贵的资源,当有多个用户访问网页的时候,对程序的性能与稳定性都有要求。数据库连接池就是为了解决这个问题的,数据库连接池负责创建、管理并释放连接。连接池中的连接可以被多个程序使用,这样就降低了创建数据库连接的开销。数据库连接池的原理也很简单,就是首先在内存存放一定数量的数据库连接,当请求的时候就从连接池中获取一个连接,该请求用完数据库连接后就会释放,于是这个连接就可以被其他程序请求。可以通过设置连接池的参数来控制连接池中的初始连接数、连接的上下限数以及每个连接的最大使用次数、最大空闲时间等等。

自己实现一个数据库连接池
目前主流的数据库连接池有dbcp、c3p0和proxool。其实现原理与上面都差不多,为了更好理解数据库连接池,我们手写一个连接池:

package cp;

import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;

public class ConnectoinPool {

    //存放Connection对象的集合
    private List<Connection> cons = null;
    private String driver = null;
    private String url = null;
    private String user = null;
    private String password = null;
    //连接池中连接的索引
    private int index = 0;
    //数据库连接池中默认大小
    private int size = 5;

    public ConnectoinPool(){
        try {
            init();
        } catch (SQLException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public ConnectoinPool(int size) {
        this.size = size;
        try {
            init();
        } catch (SQLException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    //初始化
    private void init() throws SQLException, ClassNotFoundException, IOException{
        this.cons = new ArrayList<Connection>();
        parseProperties();
        Class.forName(driver);
        for(int i=0;i<size;i++){
            Connection con = DriverManager.getConnection(url,user,password);
            cons.add(con);
        }
    }

    //读取properties文件的信息
    private void parseProperties() throws IOException {
        Properties p = new Properties();
        InputStream in = getClass().getResourceAsStream("/db.properties");
        p.load(in);
        if(p.containsKey("driver")){
            driver = p.getProperty("driver");
        }
        if(p.containsKey("url")){
            url = p.getProperty("url");
        }
        if(p.containsKey("user")){
            user = p.getProperty("user");
        }
        if(p.containsKey("password")){
            password = p.getProperty("password");
        }
    }

    //获取连接
    public Connection getConnection(){
        Connection con = cons.remove(size-1-index++);
        return con;
    }

    //释放连接
    public void release(Connection con) throws SQLException{
        cons.add(con);
        index--;
        con.close();
    }

}

创建db.properties配置文件

driver=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost/test?useUnicode=true&charset=utf8
user=root
password=1234

然后编写主程序测试各功能是否正常:

package cp;

import java.sql.Connection;
import java.sql.SQLException;
public class TestCp {
    public static void main(String[] args) throws SQLException {
        // 创建一个数据库连接池
        ConnectoinPool cp = new ConnectoinPool();
        // 获取一个连接
        Connection con1 = cp.getConnection();
        System.out.println(con1);
        //释放这个连接
        cp.release(con1);
        // 获取一个连接
        Connection con2 = cp.getConnection();
        System.out.println(con2);
        // 获取一个连接
        Connection con3 = cp.getConnection();
        System.out.println(con3);
        // 获取一个连接
        Connection con4 = cp.getConnection();
        System.out.println(con4);
        // 获取一个连接
        Connection con5 = cp.getConnection();
        System.out.println(con5);
        // 获取一个连接
        Connection con6 = cp.getConnection();
        System.out.println(con6);
    }
}

可以发现,默认连接池有5个连接对象,在释放第一个连接后,连接池仍然有5个连接,所以在程序中创建第6个连接的时候没有报错。

当然这个程序问题很多,比如不能高访问量的情况,而且是单线程的。实际情况应该是,当用户请求一个连接的时候应该创建一个线程处理,这个线程决定该连接何时释放给连接池。通过上面的简单程序,知道了连接池管理数据库连接的原理。下面我们再看看第三方的数据库连接池的使用。

Tomcat7 jdbc pool
要知道在Tomcat7以前的版本中,使用的都是dbcp连接池,但是这个连接池bug太多,Hibernate官方宣布已经不再支持dbcp连接池dbcp连接池的主要问题:

  1. dbcp 是单线程的,为了保证线程安全会锁整个连接池
  2. jdk1.6编译有问题
  3. 结构太复杂

而jdbc pool在克服以上缺点的情况下,还增加了许多优良特性,详情请看这里

其中涉及的异步方式获取连接暂时还不懂,先来学习一下如何在独立的应用中获取连接。首先需要导入Tomcat7官方给的两个jar包:
tomcat-juli.jar(Tomcat的日志模块,在bin目录下)
tomcat-jdbc.jar(jdbc pool连接池,在lib目录下)

package cp;

import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Properties;

import org.apache.tomcat.jdbc.pool.DataSource;
import org.apache.tomcat.jdbc.pool.PoolProperties;

public class TomcatJdbcPool {

    private String driver = null;
    private String url = null;
    private String user = null;
    private String password = null;
    private DataSource datasource = null;

    public TomcatJdbcPool() {
        this.datasource = new DataSource();
        try {
            parseProperties();
            initPoolProperties();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    // 读取properties文件的信息
    private void parseProperties() throws IOException {
        Properties p = new Properties();
        InputStream in = getClass().getResourceAsStream("/db.properties");
        p.load(in);
        if (p.containsKey("driver")) {
            driver = p.getProperty("driver");
        }
        if (p.containsKey("url")) {
            url = p.getProperty("url");
        }
        if (p.containsKey("user")) {
            user = p.getProperty("user");
        }
        if (p.containsKey("password")) {
            password = p.getProperty("password");
        }
    }

    private void initPoolProperties() {
        PoolProperties p = new PoolProperties();
        p.setUrl(url); // url
        p.setDriverClassName(driver); // 驱动
        p.setUsername(user); // 用户名
        p.setPassword(password); // 密码
        p.setJmxEnabled(true); // 通过JMX注册连接池
        p.setTestWhileIdle(false); // 被空闲对象使用的时候连接是否有效
        p.setTestOnBorrow(true); // 在被对象拿走之前连接是否有效
        p.setValidationQuery("SELECT 1"); // 测试是否为有效连接
        p.setTestOnReturn(false); // 连接返回连接池之前是否有效
        p.setTimeBetweenEvictionRunsMillis(30000); // 验证线程与清除线程之间的毫秒数
        p.setMaxActive(100); // 最大的活动的连接数
        p.setInitialSize(10); // 初始化连接池大小
        p.setMaxWait(10000); // 连接池等待连接返回的最长时间
        p.setRemoveAbandonedTimeout(60); // 设置废弃连接被移除之前的毫秒数
        p.setMinIdle(10); // 最小空闲连接数
        p.setLogAbandoned(true); // 标记废弃连接的堆栈跟踪
        p.setRemoveAbandoned(true); // 超时后是否可以移除连接
        p.setJdbcInterceptors("org.apache.tomcat.jdbc.pool.interceptor.ConnectionState;"
                + "org.apache.tomcat.jdbc.pool.interceptor.StatementFinalizer");
        // jdbc拦截器
        datasource.setPoolProperties(p); // 设置p到数据源中
    }

    public Connection getTomcatJdbcPoolConnection() {
        Connection con = null;
        try {
            con = datasource.getConnection(); // 通过数据源获取连接
            Statement st = con.createStatement();
            ResultSet rs = st.executeQuery("select * from user");
            int cnt = 1;
            while (rs.next()) {
                System.out.println((cnt++) + ". Host:" + rs.getString("Host")
                        + " User:" + rs.getString("User") + " Password:"
                        + rs.getString("Password"));
            }
            rs.close();
            st.close();
        } catch (SQLException e) {
            e.printStackTrace();
        } finally {
            if (con != null)
                try {
                    con.close();
                } catch (Exception ignore) {
                }
        }
        return con;
    }
}

运行结果如下:

jdbc pool 小结

使用这个数据库连接池主要就是设置相关属性,再通过数据源就可以获取连接对象了。由于进行了封装处理,所以不需要自己释放连接。这里通过properties资源文件设置数据库相关的连接属性,这样做的目的不需要修改源代码,提高程序的健壮性。

目录
相关文章
|
2月前
|
Java 关系型数据库 MySQL
"解锁Java Web传奇之旅:从JDK1.8到Tomcat,再到MariaDB,一场跨越数据库的冒险安装盛宴,挑战你的技术极限!"
【8月更文挑战第19天】在Linux上搭建Java Web应用环境,需安装JDK 1.8、Tomcat及MariaDB。本指南详述了使用apt-get安装OpenJDK 1.8的方法,并验证其版本。接着下载与解压Tomcat至`/usr/local/`目录,并启动服务。最后,通过apt-get安装MariaDB,设置基本安全配置。完成这些步骤后,即可验证各组件的状态,为部署Java Web应用打下基础。
43 1
|
4月前
|
Ubuntu 前端开发 JavaScript
技术笔记:Ubuntu:一个部署好的tomcat应用(war包)怎么用Nginx实现动静分离?
技术笔记:Ubuntu:一个部署好的tomcat应用(war包)怎么用Nginx实现动静分离?
|
2月前
|
SQL Java 关系型数据库
探索Java数据库连接的奥秘:JDBC技术全攻略
探索Java数据库连接的奥秘:JDBC技术全攻略
47 8
|
1月前
|
关系型数据库 Java MySQL
"解锁Java Web传奇之旅:从JDK1.8到Tomcat,再到MariaDB,一场跨越数据库的冒险安装盛宴,挑战你的技术极限!"
【9月更文挑战第6天】在Linux环境下安装JDK 1.8、Tomcat和MariaDB是搭建Java Web应用的关键步骤。本文详细介绍了使用apt-get安装OpenJDK 1.8、下载并配置Tomcat,以及安装和安全设置MariaDB(MySQL的开源分支)的方法。通过这些步骤,您可以快速构建一个稳定、高效的开发和部署环境,并验证各组件是否正确安装和运行。这为您的Java Web应用提供了一个坚实的基础。
38 0
|
2月前
|
jenkins 持续交付 开发工具
自动化开发之旅:Docker携手Jenkins,与Git和Tomcat共舞持续集成
【8月更文挑战第13天】在软件开发中,持续集成(CI)通过自动化构建、测试与部署提升效率与稳定性。Docker、Jenkins、Git和Tomcat构成CI的黄金组合:`git push`触发Jenkins作业,利用Docker确保环境一致性,最终将应用部署至Tomcat。首先配置Git Webhooks以触发Jenkins;接着在Jenkins中创建作业并使用Docker插件模拟真实环境;通过Maven构建项目;最后部署至Tomcat。这套流程减少人为错误,提高开发效率,展示了技术的力量与流程的革新。
62 0
|
3月前
|
算法 Java 应用服务中间件
开发与运维机制问题之在Tomcat的类加载机制中,如果BootstrapClassLoader没有加载成功类,Tomca如何解决
开发与运维机制问题之在Tomcat的类加载机制中,如果BootstrapClassLoader没有加载成功类,Tomca如何解决
18 0
|
3月前
|
Java 应用服务中间件 API
开发与运维机制问题之Tomcat要打破双亲委派机制如何解决
开发与运维机制问题之Tomcat要打破双亲委派机制如何解决
22 0
|
4月前
|
存储 设计模式 搜索推荐
早期javeweb技术 JSP JDBC JSTJ Servlet BooStrap(下)
早期javeweb技术 JSP JDBC JSTJ Servlet BooStrap(下)
25 1
|
4月前
|
Java 数据库连接 API
后端开发之用Mybatis简化JDBC的开发快速入门2024及数据库连接池技术和lombok工具详解
后端开发之用Mybatis简化JDBC的开发快速入门2024及数据库连接池技术和lombok工具详解
55 3
|
3月前
|
Java 应用服务中间件 Linux
Tomcat安装部署[单机软件],可以让用户开发的WEB应用程序,变成可以被访问的网页,Tomcat的使用需要jdk环境
Tomcat安装部署[单机软件],可以让用户开发的WEB应用程序,变成可以被访问的网页,Tomcat的使用需要jdk环境