J2EE开发技术点5:Tomcat jdbc pool

2024-05-30 17:08

本文主要是介绍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); // urlp.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资源文件设置数据库相关的连接属性,这样做的目的不需要修改源代码,提高程序的健壮性。

这篇关于J2EE开发技术点5:Tomcat jdbc pool的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1016575

相关文章

最新版IDEA配置 Tomcat的详细过程

《最新版IDEA配置Tomcat的详细过程》本文介绍如何在IDEA中配置Tomcat服务器,并创建Web项目,首先检查Tomcat是否安装完成,然后在IDEA中创建Web项目并添加Web结构,接着,... 目录配置tomcat第一步,先给项目添加Web结构查看端口号配置tomcat    先检查自己的to

Apache Tomcat服务器版本号隐藏的几种方法

《ApacheTomcat服务器版本号隐藏的几种方法》本文主要介绍了ApacheTomcat服务器版本号隐藏的几种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需... 目录1. 隐藏HTTP响应头中的Server信息编辑 server.XML 文件2. 修China编程改错误

若依部署Nginx和Tomcat全过程

《若依部署Nginx和Tomcat全过程》文章总结了两种部署方法:Nginx部署和Tomcat部署,Nginx部署包括打包、将dist文件拉到指定目录、配置nginx.conf等步骤,Tomcat部署... 目录Nginx部署后端部署Tomcat部署出现问题:点击刷新404总结Nginx部署第一步:打包

Nginx、Tomcat等项目部署问题以及解决流程

《Nginx、Tomcat等项目部署问题以及解决流程》本文总结了项目部署中常见的four类问题及其解决方法:Nginx未按预期显示结果、端口未开启、日志分析的重要性以及开发环境与生产环境运行结果不一致... 目录前言1. Nginx部署后未按预期显示结果1.1 查看Nginx的启动情况1.2 解决启动失败的

tomcat在nginx中的配置方式

《tomcat在nginx中的配置方式》文章介绍了如何在Linux系统上安装和配置Tomcat,并通过Nginx进行代理,首先,下载并解压Tomcat压缩包,然后启动Tomcat并查看日志,接着,配置... 目录一、下载安装tomcat二、启动tomcat三、配置nginx总结提示:文章写完后,目录可以自动

滚雪球学Java(87):Java事务处理:JDBC的ACID属性与实战技巧!真有两下子!

咦咦咦,各位小可爱,我是你们的好伙伴——bug菌,今天又来给大家普及Java SE啦,别躲起来啊,听我讲干货还不快点赞,赞多了我就有动力讲得更嗨啦!所以呀,养成先点赞后阅读的好习惯,别被干货淹没了哦~ 🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,助你一臂之力,带你早日登顶🚀,欢迎大家关注&&收藏!持续更新中,up!up!up!! 环境说明:Windows 10

详解Tomcat 7的七大新特性和新增功能(1)

http://developer.51cto.com/art/201009/228537.htm http://tomcat.apache.org/tomcat-7.0-doc/index.html  Apache发布首个Tomcat 7版本已经发布了有一段时间了,Tomcat 7引入了许多新功能,并对现有功能进行了增强。很多文章列出了Tomcat 7的新功能,但大多数并没有详细解释它们

Tomcat性能参数设置

转自:http://blog.csdn.net/chinadeng/article/details/6591542 Tomcat性能参数设置 2010 - 12 - 27 Tomcat性能参数设置 博客分类: Java Linux Tomcat 网络应用 多线程 Socket 默认参数不适合生产环境使用,因此需要修改一些参数   1、修改启动时内存参数、并指定J

TL-Tomcat中长连接的底层源码原理实现

长连接:浏览器告诉tomcat不要将请求关掉。  如果不是长连接,tomcat响应后会告诉浏览器把这个连接关掉。    tomcat中有一个缓冲区  如果发送大批量数据后 又不处理  那么会堆积缓冲区 后面的请求会越来越慢。

Hibernate框架中,使用JDBC语法

/*** 调用存储过程* * @param PRONAME* @return*/public CallableStatement citePro(final String PRONAME){Session session = getCurrentSession();CallableStatement pro = session.doReturningWork(new ReturningWork<C