介绍
在使用Java语言进行和数据库有关的的应用开发中,一般都使用JDBC来进行和数据库的交互,其中有一个关键的概念就是Connection(连接),它在Java中是一个类,代表了一个通道。通过它,使用数据的应用就可以从数据库访问数据了。
对于一个简单的数据库应用,由于对于数据库的访问不是很频繁。这时可以简单地在需要访问数据库时,就新创建一个连接,用完后就关闭它,这样做也不会带来什么明显的性能上的开销。但是对于一个复杂的数据库应用,情况就完全不同了。频繁的建立、关闭连接,会极大的减低系统的性能,因为对于连接的使用成了系统性能的瓶颈。
本文给出的方法可以有效的解决这个问题。在本方法中提出了一个合理、有效的连接管理策略,避免了对于连接的随意、无规则的使用。该策略的核心思想是:连接复用。通过建立一个数据库连接池以及一套连接使用管理策略,使得一个数据库连接可以得到高效、安全的复用,避免了数据库连接频繁建立、关闭的开销。另外,由于对JDBC中的原始连接进行了封装,从而方便了数据库应用对于连接的使用(特别是对于事务处理),提高了开发效率,也正是因为这个封装层的存在,隔离了应用的本身的处理逻辑和具体数据库访问逻辑,使应用本身的复用成为可能。
连接池模型
本文讨论的连接池包括一个连接池类(DBConnectionPool)和一个连接池管理类 (DBConnetionPoolManager)。连接池类是对某一数据库所有连接的“缓冲池”,主要实现以下功能:
①从连接池获取或创建可用连接;
② 使用完毕之后,把连接返还给连接池;
③在系统关闭前,断开所有连接并释放连接占用的系统资源;
④还能够处理无效连接(原来登记为可用的连接,由于某种原因 不再可用,如超时,通讯问题),并能够限制连接池中的连接总数不低于某个预定值和不超过某个预定值。
连接池管理类是连接池类的外覆类(wrapper),符合单例模式,即系统中只能有一个连接池管理类的实例。其主要用于对多个连接池对象的管 理,具有以下功能:
①装载并注册特定数据库的JDBC驱动程序;
②根据属性文件给定的信息,创建连接池对象;
③为方便管理多个连接池对象,为每一个连接池 对象取一个名字,实现连接池名字与其实例之间的映射;
④跟踪客户使用连接情况,以便需要是关闭连接释放资源。
连接池管理类的引入主要是为了方便对多个连接池的使用和管理,如系统需要连接不同的数据库,或连接相同的数据库但由于安全性问题,需要不同的用户使用不同的名称和密码。
连接池实现
下面给出连接池类和连接池管理类的主要属性及所要实现的基本接口:
- public class DBConnectionPool implements TimerListener{
- private int checkedOut;//已被分配出去的连接数
- private ArrayList freeConnections=new ArrayList();
- //容器,空闲池,根据//创建时间顺序存放已创建但尚未分配出去的连接
- private int minConn;//连接池里连接的最小数量
- private int maxConn;//连接池里允许存在的最大连接数
- private String name;//为这个连接池取个名字,方便管理
- private String password;//连接数据库时需要的密码
- private String url;//所要创建连接的数据库的地址
- private String user;//连接数据库时需要的用户名
- public Timer timer;//定时器
- public DBConnectionPool(String name,String URL,String user,
- String password,int maxConn)//公开的构造函数
- public synchronized void freeConnection(Connection con)
- //使用完毕之后,把连接返还给空闲池
- public synchronized Connection getConnection(long timeout)
- //得到一个连接,timeout是等待时间
- public synchronized void release()
- //断开所有连接,释放占用的系统资源
- private Connection newConnection()
- //新建一个数据库连接
- public synchronized void TimerEvent()
- //定时器事件处理函数
- }
- public class DBConnectionManager {
- static private DBConnectionManager instance;
- //连接池管理类的唯一实例
- static private int clients;//客户数量
- private ArrayList drivers=new ArrayList();
- //容器,存放数据库驱动程序
- private HashMap pools = new HashMap();
- //以name/value的形式存取连接池对象的名字及连接池对象
- static synchronized public DBConnectionManager getInstance()
- /**如果唯一的实例instance已经创建,直接返回这个实例;否则,调用私有构造函数,
- 创建连接池管理类的唯一实例*/
- private DBConnectionManager()
- //私有构造函数,在其中调用初始化函数init()
- public void freeConnection(String name,Connection con)
- //释放一个连接,name是一个连接池对象的名字
- public Connection getConnection(String name)
- //从名字为name的连接池对象中得到一个连接
- public Connection getConnection(String name,long time)
- //从名字为name的连接池对象中取得一个连接,time是等待时间
- public synchronized void release()//释放所有资源
- private void createPools(Properties props)
- //根据属性文件提供的信息,创建一个或多个连接池
- private void init()//初始化连接池管理类的唯一实例,由私有构造函数调用
- private void loadDrivers(Properties props)//装载数据库驱动程序
- }
连接池使用
上面所实现的连接池在程序开发时如何应用到系统中呢?下面以Servlet为例说明连接池的使用。
Servlet的生命周期是:在开始建立servlet时,调用其初始化(init)方法。之后每个用户请求都导致一个调用前面建立的实例的service方法的线程。最后,当服务器决定卸载一个servlet时,它首先调用该servlet的destroy方法。
根据servlet的特点,我们可以在初始化函数中生成连接池管理类的唯一实例(其中包括创建一个或多个连接池)。如:
- public void init() throws ServletException
- {
- connMgr=DBConnectionManager.getInstance();
- }
- 然后就可以在service方法中通过连接池名称使用连接池,执行数据库操作。最后在destroy方法中释放占用的系统资源,如:
- public void destroy(){
- connMgr.release();
- super.destroy();
- }
结束语
在使用JDBC进行与数据库有关的应用开发中,数据库连接的管理是一个难点。很多时候,连接的混乱管理所造成的系统资源开销过大成为制约大型企 业级应用效率的瓶颈。对于众多用户访问的Web应用,采用数据库连接技术的系统在效率和稳定性上比采用传统的其他方式的系统要好很多。本文阐述了使用 JDBC访问数据库的技术?讨论了基于连接池技术的数据库连接管理的关键问题并给出了一个实现模型。文章所给出的是连接池管理程序的一种基本模式,为提高 系统的整体性能,在此基础上还可以进行很多有意义的扩展。