数据库连接池的设计考虑
周末检到本j2ee的书翻了两下(1200多页,我靠),其中有提到jdbc的连接池,小有兴趣,因为我的vc-ado程序也可以考虑实现个简单连接池来用用。
ado的连接有几个特点:
1.它不是线程安全的。容易造成事务死锁。一般单条sql的不会死锁,但比如执行事务或游标时就很容易死锁掉.所以一般的多线程程序不提倡重用同一个ado connection
2:ado会自己管理oledb连接池,当调用connection的close方法,仅仅是将连接放到池里,直到超时或程序关闭。一般来说,一个程序使用相同连接串创建的连接都会被自动池化。ado.net可以明确的选择是否使用池。
3:如果使用智能指针_connectionprt,其close方式只是关闭连接(放到池),而没有释放对象。
4:但如果一直不调用close,该连接则无法被其他线程/进程共享
5:在某个连接上忙,且程序要求在该连接上执行操作的时候,ado其实会打开一个临时连接供使用。
之所以考虑ado的连接池,主要是执行一些简单的sql时(如insert,delete..),不需要多创建很多ado connection对象。
另外也随便记录一点设计一个池要考虑的一些要素:
1:池化。包括批次初始化和释放连接。可以设计成忙/闲队列管理,一有调用,就从闲队列里取出一个,并将其引用放到忙队列里;或者采用循环数组,通过记数管理。倾向于使用后者,因为前者最多的连接数不能超出队列大小,而实际上每个连接是可以被客户端复用的。记数管理的目的是获取连接池里被引用最少的一个连接。
2:给客户端统一的获取和释放接口。pool.getconnection这个好理解。也可以定义一个pool.releaseconnection(connection)接口,但客户端代码难免会习惯性的调用connection.close(即使禁止)。最好的办法就是实现或者说重载connection接口,接管或屏蔽close.使close的默认动作是将连接返回pool里待用(或者减少在连接上的使用记数).
3:检测连接上的事务完成状态。在一个复用的连接上交叉执行事务是不应该的。在一个事务没有执行完毕时,不允许加入另一个事务(通过进程内锁实现)。但目前编写的代码基本上没有使用到transaction的,所以不着急。
4:从2可以看到,最好是定义一个pooledconnection接口,客户端对接口实例操作,而不是直接对connection对象操作,这样才能更好的实现池化的一些控制逻辑。但这样又牵涉到接口和对象间的同步维护。最好是用继承的方法实现,除了打开关闭等直接影响连接状况的接口外,还是给客户端对对象的大部分直接控制接口。
5:连接池应检测connection的连接状态,断连的连接应该被标记或直接移出连接池。