持久层设计与资源管理模式
无论是怎样的应用系统,都无法脱离对资源的管理和使用。而对于持久层而言,资源的合理管理和调度则显得尤为重要。
在大多数应用系统中,80%以上的应用逻辑并不需要特别复杂的数据库访问逻辑(可能只是几条简单的Select或作者Insert/Update语句)。对于这些占到多数的简单逻辑而言,如果SQL语句和数据库本身的设计不是太糟糕(合适的关联,字段索引以及数据库分区策略),在特定的硬件环境下,我们认为数据库的性能基本稳定。
此时,导致持久层性能地下的罪魁祸首可能并不是数据库本身,而是在于失败的资源管理调度机制。
Connection Pool
即使对于我们而言,通过JDBC获取数据库连接实在是件再简单不过的事情,但对于JDBC Driver来说,连接数据库却并非一件轻松差事。数据库连接不仅仅是在应用服务器与数据库服务器之间建立一个Socket Connection(对于Type4 的JDBC Driver而言),连接建立之后,应用服务器和数据库服务器之间还需要交换若干次数据(验证用户密码,权限等),然后,数据库开始初始化连接会话句柄,记录联机日志,为此连接分配相应得处理进程和系统资源。
系统如此忙碌,如果我们只是简单地仍过去两个SQL语句,然后就将此连接抛弃,实在可惜,而数据库连接池技术正是为了解决这个问题。
数据库连接池的基本原理是在内部对象池中维护一定数量的数据库连接,并对外暴露数据库连接获取或返回方法。如图
数据库连接池基本原理示意
外部使用者可以通过getConnection方法获取连接,使用完毕后再通过releaseConnection方法将连接返回,注意此时连接并没有关闭,而是由连接池管理器回收,并为下一次使用做好准备。
数据连接池技术带来下面的优势:
1. 资源重用
由于数据库连接得以重用,避免了频繁创建,释放连接引起的大量性能开销。在减少系统消耗的基础上,另一方面也增进了系统运行环境的平稳性(减少内存碎片以及数据库临时进程/线程的数量)。
2. 更快的系统响应速度
数据库连接池仔初始化过程中,往往已经创建了若干数据库连接池置于池中备用。此时连接的初始化工作均已完成。对于业务请求处理而言,直接利用现有可用连接,避免了数据库连接初始化和释放过程的时间开销,从而缩减了系统整体响应时间。
3. 新的资源分配手段
对于应用共享同一个数据库的系统而言,可在应用曾通过数据库连接池的配置,实现某一应用最大可用数据库连接数的限制,避免某一应用独占所有数据库资源。
4. 统一的连接管理,避免数据库连接泄露
在较为完备的数据库连接池实现中,可根据预先的连接占用超时设定,强制收回被占用连接。从而避免了常规数据库连接操作中可能出现的资源泄露。
一个最小化的数据库连接池实现:
package net.wanjin.lab.utils;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.util.Vector;
public class ConnectionPool {
private static Vector pool;
private final int POOL_MAX_SIZE =20;
/**
* 获取数据库连接
* 如果当前池中有可用连接,则将池中最后一个返回,如果没有,则新建一个返回
* @return
* @throws DBException
*/
public synchronized Connection getConnection() throws DBException{
if(pool ==null){
pool = new Vector();
}
Connection conn;
if(pool.isEmpty()){
conn = createConnection();
}else{
int last_idx = pool.size() -1;
conn =(Connection)pool.get(last_idx);
pool.remove(pool.get(last_idx));
}
return conn;
}
/**
* 将使用完毕的数据库连接放回备用池
*
* 判断当前池中连接数是否已经超过阀值
* 如果超过,则关闭该连接
* 否则放回池中以备下次重用
* @param conn
*/
public synchronized void releaseConnection(Connection conn){
if(pool.size()>POOL_MAX_SIZE){
try{
conn.close();
}catch(SQLException e){
e.printStackTrace();
}
}else{
pool.add(conn);
}
}
/**
* 读取数据库配置信息,并从数据库连接池中获得数据库连接
* @return
* @throws DBException
*/
private static Connection createConnection() throws DBException{
Connection conn;
try{
Class.forName("oracle.jdbc.driver.OracleDriver");
conn = DriverManager.getConnection("jdbc:oracle:thin:@1521:oracle","personal","personal");
return conn;
}catch(ClassNotFoundException e){
throw new DBException("ClassNotFoundException when loading JDBC Driver");
}catch(SQLException e){
throw new DBexception("SQLException when loading JDBC Driver");
}
}
}
完备的连接池实现固然错综复杂,但就其根本而言,还是源自同样的思想。
先脱离连接池本身的具体实现,我们看看这段代码在实际应用中可能产生的问题,注意到,由于getConnection方法返回的是一个标准的JDBC Connection,程序员由于编成习惯,可能会习惯性的调用其close方法关闭连接。如此一来,连接无法得到重用,数据库连接池机制形同虚设。为了解决这个问题,比较好的途径有:
1. Decorator 模式
2. Dynamic Proxy 模式
下面我们就这两种模式进行一些探讨:
“Decorator模式的主要目的是利用一个对象,透明地为另一个对象添加新的功能”。
这句话是从GOF关于设计模式的经典著作《设计模式-可复用面向对象软件的基础》一书中关于Decorator模式的描述直译而来,可能比较难以理解。简单来讲,就是通过一个Decorator对原有对象进行封装,同时实现与原有对象相同的接口,从而得到一个基于原有对象的,对既有接口的增强性实现。
对于前面所讨论的Connection释放的问题,理所当然,我们首先想到的是,如果能让JDBC Connection在执行close操作时自动返回到数据库连接池中,那么所有问题都将迎刃而解,但是,JDBC Connection自己显然无法根据实际情况判断何去何从。此时,引入Decorator模式来解决我们所面对的问题就非常合适。
首先,我们引入一个ConnectionDecorator类:
import java.sql.Connection;
import java.sql.SQLException;
public class ConnectionDecorator implements Connection {
Connection dbconn;
public ConnectionDecorator(Connection conn){
this.dbconn = conn; //实际从数据库获得的Connection引用
}
/**
* 此方法将被子类覆盖,以实现数据库连接池的连接返回操作
* @see java.sql.Connection#close()
*/
public void close() throws SQLException{
this.dbconn.close();
}
/**
* (non-Javadoc)
* @see java.sql.Connectionn#commit()
*/
public void commit() throws SQLException{
this.dbconn.commit(); //调用实际连接的commit方法
}
//以下各个方法类似,均调用dbconn.xxx方法作为Connection接口中方法的简单实现.
}
可以看到,ConnectionDecorator类实际上的对传入的数据库连接加上一个外壳,他实现了java.slq.Connection接口,不过本身并没有实现任何实际内容,只是简单的把方法的实现委托给运行期实际获得的Connection实例,,而从外部来看,ConnctionDecorator与普通Connection实例并没有什么区别,因为他们实现了同样的接口,对外提供了同样的功能调用。
目标很清楚,通过这样的封装,我们的ConnectionDecorator对于外部的程序员而言,调用方法与普通的JDBC Connection完全相同,而在内部通过对ConnectionDecorator的修改,我们就可以透明地改变现有实现,为了增加新的特性:
package net.wanjin.lab.persistence.Decorator;
import java.sql.Connection;
import net.wanjin.lab.utils.ConnectionPool;
public class PooledConnection extends ConnectionDecorator implements Connection{
private ConnectionPool connPool;
public PooledConnection(ConnectionPool pool,Connection conn){
super(conn);
connPool = pool;
}
/**
* 覆盖Close方法,将数据库连接返回连接池,而不是直接关闭连接
*/
public void close() throws SQLException{
connPool.releaseConnection(this.dbconn);
}
}
为了应用新的PooledConnection ,我们需要对原本的DBConnectionPool.getConnection和releaseConnection方法稍做改造:
/**
* 获取数据库连接
* 如果当前池中有可用连接,则将池中最后一个返回,如果没有,则新建一个返回
* @return
* @throws DBException
*/
public synchronized Connection getConnection() throws DBException{
if(pool ==null){
pool = new Vector();
}
Connection conn;
if(pool.isEmpty()){
conn = createConnection();
}else{
int last_idx = pool.size() -1;
conn =(Connection)pool.get(last_idx);
pool.remove(pool.get(last_idx));
}
//return conn;
return new PooledConnection(this,conn);
}
/*
* 将使用完毕的数据库连接放回备用池
*
* 判断当前池中连接数是否已经超过阀值
* 如果超过,则关闭该连接
* 否则放回池中以备下次重用
* @param conn
*/
public synchronized void releaseConnection(Connection conn){
if(conn instanceof PooledConnection||pool.size()>POOL_MAX_SIZE){
try{
conn.close();
}catch(SQLException e){
e.printStackTrace();
}
}else{
pool.add(conn);
}
}
此时获取数据库连接后,调用这只需要按照JDBC Connection的标准用法进行调用即可,从而实现了数据库连接的透明化。同样的道理,我们甚至可以利用Decorator模式对DriverManager类进行同样的改造,从而最小化数据库连接池实现对传统JDBC编码方式的影响。
Dynamic Proxy 模式
DynamicProxy是JDK1.3版本中新引入的一种代理机制。严格来讲,DynamicProxy本身并非一种模式,只能算是Proxy模式的一种动态实现方式,不过为了与传统Proxy模式相区分,这里暂且将其称为“Dynamic Proxy模式”,来泛指通过Dynamic Proxy机制实现的Proxy模式。
回到上面的问题,我们的目标是在引入数据库连接池机制的同时,保持JDBCConnection对外接口不变。前面通过Decorator接口中定义的方法众多,我们也只是能照单全收,在ConnecionDecorator中逐一实现这些方法,虽然只是简单的委托实现,也实在是件恼人的工作。
Dynamic Proxy模式则良好地解决了这一问题。通过实现一个邦定到Connection对象的InvocationHandler接口实现,我们可以在Connection.close方法被调用时将其截获,并以我们自己实现的Close方法将其代替,使连接返回到数据库连接池等待下次重用,而不是直接关闭。
下面是ConnectionHandler类,它实现了InvocationHandler接口,按照DynamicProxy机制的定义,Invoke方法将截获所有代理对象的方法调用操作,这里我们通过invoke方法截获close方法进行处理。
package net.wanjin.lab.utils;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.sql.Connection;
public class ConnectionHandler implements InvocationHandler {
Connection dbconn;
ConnectionPool pool;
public ConnectionHandler(ConnectionPool connPool){
this.pool = connPool;
}
/**
*
*/
public Connection bind(Connection conn){
this.dbconn = conn;
Connection proxyConn =(Connection)Proxy.newProxyInstance(
conn.getClass().getClassLoader(),conn.getClass().getInterfaces(),this);
return proxyConn;
}
public Object invoke(Object proxy,Method method,Object[] args)
throws Throwable{
Object obj =null;
//如果调用的是close方法,则用pool.releaseConnection方法将其替换
if("close".equals(method.getName())){
pool.releaseConnection(dbconn);
}else{
obj = method.invoke(dbconn, args);
}
return obj;
}
}
配合Dynamic Proxy模式,我们的DBConnectionPool.getConnection方法也做了一点小的修改:
/**
* 获取数据库连接
* 如果当前池中有可用连接,则将池中最后一个返回,如果没有,则新建一个返回
* @return
* @throws DBException
*/
public synchronized Connection getConnection() throws DBException{
if(pool ==null){
pool = new Vector();
}
Connection conn;
if(pool.isEmpty()){
conn = createConnection();
}else{
int last_idx = pool.size() -1;
conn =(Connection)pool.get(last_idx);
pool.remove(pool.get(last_idx));
}
//return conn;
//return new PooledConnection(this,conn);
ConnectionHandler connHandler = new ConnectionHandler(this);
return connHandler.bind(conn);
}
可以看到,基于Dynamic Proxy 模式的实现相对Decorator更加简洁明了。
分享到:
相关推荐
Hibernate使用了一种称为“一级缓存”的资源管理模式,每个Session都有自己的缓存,保存了最近访问过的对象。此外,还有二级缓存,它是一种可选的全局缓存,可以跨多个Session共享。 ### 解耦合设计 通过ORM,...
MyBatis则是一个持久层框架,允许开发者用SQL语句直接操作数据库,同时提供了映射机制,将Java对象与数据库表中的记录对应起来,简化了数据访问层的开发。 在"在线云笔记"项目中,Spring作为整体架构的中心,负责...
1. **myBatis**:myBatis是一个优秀的持久层框架,它支持定制化SQL、存储过程以及高级映射。myBatis避免了几乎所有的JDBC代码和手动设置参数以及获取结果集。它将接口和Java的POJOs(Plain Old Java Objects,普通的...
P2P架构与之不同,它是一种去中心化的网络结构,在这种模式下,每个参与节点既是资源的提供者也是请求者。P2P网络通常用于文件共享、分布式计算等场景,它具有较强的容错性和扩展性,但管理和监控难度较大。 文件中...
**JAVA人力资源管理系统设计文档** 在信息技术日新月异的今天,JAVA作为一种强大的、跨平台的编程语言,被广泛应用于各种复杂系统的开发,其中包括人力资源管理系统(HRM System)。本设计文档详细阐述了如何利用...
MyBatis是一个轻量级的持久层框架,允许开发者直接编写SQL语句,将数据库查询结果映射到Java对象,降低了数据访问的复杂性。 MyBatis与Spring的整合使得数据库操作更加灵活,开发者可以根据需求编写自定义的SQL,...
MyBatis 是一款优秀的持久层框架,它支持定制化 SQL、存储过程以及高级映射。MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。MyBatis 可以使你更好地将关注点集中在你的应用程序上。同时,它也...
Struts2是MVC(模型-视图-控制器)设计模式的一个实现,它为Java Web开发提供了强大的控制层。Struts2的核心是Action类,它处理用户请求并调用业务逻辑。以下是一些关键知识点: 1. **FilterDispatcher**: Struts2...
4. **MyBatis**:MyBatis是一个优秀的持久层框架,它支持定制化SQL、存储过程以及高级映射。MyBatis避免了几乎所有的JDBC代码和手动设置参数以及获取结果集。它使开发者能够更好地关注SQL语句本身,而不必关注JDBC的...
10. **MyBatis**:MyBatis是一个轻量级的持久层框架,它简化了SQL与Java代码的绑定,提高了数据库操作的灵活性。 11. **Hibernate**:Hibernate是一个ORM(对象关系映射)框架,它允许开发者以面向对象的方式处理...
3. **20、MyBatis.pdf**:MyBatis是一个优秀的持久层框架,讲解了如何通过XML或注解的方式将SQL与Java代码绑定,简化数据库操作,以及MyBatis的动态SQL和事务管理。 4. **6、JavaSE:面向对象.pdf**:这部分深入...
MyBatis是一个持久层框架,它简化了Java与数据库之间的交互,避免了传统JDBC代码的繁琐。MyBatis允许开发者编写SQL语句,将它们映射到Java方法上,实现了动态SQL。通过Mapper接口,可以方便地进行CRUD操作,并且支持...
SSM框架中的Spring负责依赖注入和管理,SpringMVC处理HTTP请求和视图解析,MyBatis则作为持久层框架,简化数据库操作。 【标签】"java ssm"进一步确认了核心主题,即Java语言和SSM框架的学习与实践。Java是一种广泛...
- **MyBatis**:简化SQL操作的持久层框架,将SQL与Java代码分离。 以上只是部分Java学习中的关键知识点,实际的"java各种笔记.zip"文件可能涵盖更多内容,如JSP、Servlet、JavaFX、Java 8新特性等。这些笔记对于...
MyBatis是一个持久层框架,它允许开发者用简单的XML或注解来配置和映射原生信息,将接口和Java的POJOs(Plain Old Java Objects,普通的Java对象)映射成数据库中的记录。在本项目中,MyBatis负责与MySQL数据库的交互...
6. **Java Persistence API(JPA)与Hibernate**:JPA是Java EE规范,用于管理和持久化Java对象。Hibernate是JPA的一个实现,它提供了强大的ORM(对象关系映射)功能,简化了数据库操作。 7. **Spring框架**:...
Mybatis是一个轻量级的持久层框架,它简化了JDBC的复杂性,提供了更加灵活的SQL编写方式。在笔记中,可能会详细讲解Mybatis的配置文件、Mapper接口和XML配置,以及如何使用注解进行映射。还会涉及动态SQL,例如if、...
MyBatis则是一个轻量级的持久层框架,它简化了SQL操作,将SQL语句与Java代码紧密结合,避免了传统的JDBC代码繁琐的编写过程。 在SSM框架整合中,首先需要配置Spring的Bean定义,这包括数据库连接、事务管理等。然后...
MyBatis是一个轻量级的持久层框架,它允许开发者使用SQL语句进行数据库操作,同时提供了动态SQL的支持。在【note_day12_all】中,可能会涉及MyBatis的XML配置文件、Mapper接口的创建和使用,以及如何通过注解方式...
Mybatis,作为一款优秀的持久层框架,它简化了Java与数据库之间的交互,通过XML或注解方式配置和映射原生信息,将接口和Java的POJOs(Plain Old Java Objects,普通的Java对象)映射成数据库中的记录。在mybatisDay01...