很多人都对基于JDBC的数据库应用开发中的数据库连接的管理头疼,因为它是决定该应用性能的一个重要 因素。 本文在对数据库连接进行透彻分析的基础上,提出并实现了一个高效的连接管理策略,

使得开发高性能的数据库应用变得相对容易。特别是,对于连接管理中的两个 难点:事务和多线程问题进行了深入的剖析,并给出了一个基于设计模式的解决方案。
介绍

在使用Java语言进行和数据库有关的的应用开发中,一般都使用JDBC来进行和数据库的交互,其中有一个关键的概念就是Connection(连接),它在Java中是一个类,代表了一个通道。通过它,使用数

据的应用就可以从数据库访问数据了。

对 于一个简单的数据库应用,由于对于数据库的访问不是很频繁。这时可以简单地在需要访问数据库时,就新创建一个连接,用完后就关闭它,这样做也不会带来什么 明显的性能上的开销。但是对于

一个复杂的数据库应用,情况就完全不同了。频繁的建立、关闭连接,会极大的减低系统的性能,因为对于连接的使用成了系统性能 的瓶颈。

本文给出的方法可以有效的解决这个问题。在本方法中提出了一个合理、有效的连接管理策略,避免了对于连接的随意、无规则的使用。 该策略的核心思想是:连接复用。通过建立一个数据库连接池以

及一套连接使用管理策略,使得一个数据库连接可以得到高效、安全的复用,避免了数据库连接频繁 建立、关闭的开销。另外,由于对JDBC中的原始连接进行了封装,从而方便了数据库应用对于连接的

使用(特别是对于事务处理),提高了开发效率,也正是因 为这个封装层的存在,隔离了应用的本身的处理逻辑和具体数据库访问逻辑,使应用本身的复用成为可能。

问题产生

[url=http://www.bjtarena.com/][color=black]达内[/color][/url]参与的项目是开发一个网管系统,不可避免的要和数据库打交道。刚开始时,由于对于数据库的访问不是很频繁,对于数据库连接的使用就是简单的需要时就建立,用完就关 闭的策略,这很符合XP

(eXtreme Programming)的口号:”Do the Simplest Thing that Could Possibly Work”。确实,开始时工作的很好。随着项目的进展,对于数据库的访问开始变的频繁,问题就暴露出来了,原先的

通过简单地获取和关闭数据库连接的方法将 很大的影响系统的性能,这种影响是由于数据库资源管理器进程频繁的创建和摧毁那些连接对象而引起的。

此时,就有必要对数据库访问方法进行重构(refactoring),因为我们确实需要进行改进,来提高系统的性能。

解决方案

可 以看出,问题的根源就是由于对于连接资源的低效管理造成的。对于共享资源,有一个很著名的设计模式:资源池。该模式正是为了解决资源频繁分配、释放所造成 的问题的。把该模式应用到数据

库连接管理领域,就是建立一个数据库连接池,提供一套高效的连接分配、使用策略,最终目标是实现连接的高效、安全的复用。

1.1、建立连接池

第 一步,就是要建立一个静态的连接池,所谓静态是指,池中的连接是在系统初始化时就分配好的,并且不能够随意关闭的。Java中给我们提供很多容器类可以方 便的用来构建连接池,如:Vector、

Stack等。在系统初始化时,根据配置创建连接并放置在连接池中,以后所使用的连接都是从该连接池中获取的,这 样就可以避免连接随意建立、关闭造成的开销(当然,我们没有办法避免Java的

Garbage Collection带来的开销)。

1.2、分配、释放策略

有了这个连接池,下面我们就可以提供一套自定义的分配、释放策略。

当 客户请求数据库连接时,首先看连接池中是否有空闲连接,这里的空闲是指,目前没有分配出去的连接。如果存在空闲连接则把连接分配给客户,并作相应处理,具 体处理策略,在关键议题中会详

述,主要的处理策略就是标记该连接为已分配。若连接池中没有空闲连接,就在已经分配出去的连接中,寻找一个合适的连接给客户 (选择策略会在关键议题中详述),此时该连接在多个客户间复用。

当客户释放数据库连接时,可以根据该连接是否被复用,进行不同的处理。如果连接没有使用者,就放入到连接池中,而不是被关闭。

可以看出正是这套策略保证了数据库连接的有效复用。

1.3、配置策略

数据库连接池中到底要放置多少个连接,连接耗尽后该如何处理呢?这时一个配置策略。一般的配置策略是,开始时,根据具体的应用需求,给出一个初始的连接池中连接的数目以及一个连接池可以扩

张到的最大连接数目。本方案就是按照这种策略实现的。

关键议题


2.1、引用记数

1.2 节中的分配、释放策略对于有效复用连接非常重要,我们采用的方法也是采用了一个很有名的设计模式:Reference Counting(引用记数)。该模式在复用资源方面用的非常广泛,我们把该方法运

用到对于连接的分配释放上。每一个数据库连接,保留一个引用记数,用来 记录该连接的使用者的个数。具体的实现上,我们采用了两极连接池,空闲池和使用池

本节将对上述解决方案中的关键细节进行详述,正是这些关键的策略保证了数据库连接复用的高效和安全。