在上篇文章中《代理模式之静态代理---数据库连接池对象实现原理》中介绍了使用静态代理实现数据库连接池在关闭数据库链接对象的问题。
其实在JDK1.3以后就提供了动态代理的方式。使用动态代理解决上面提到的问题会更简便一些。
因此我们就需要修改数据源类MyDataSource类的创建数据库链接对象的方法createConnection方法如下,其他则不用修改
其实在JDK1.3以后就提供了动态代理的方式。使用动态代理解决上面提到的问题会更简便一些。
package com.zzg.jdbc.datasource;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.sql.Connection;
public class MyConnectionHandler implements InvocationHandler {
private MyDataSource myds;
private Connection realConn;
private Connection proxyConn;
MyConnectionHandler(MyDataSource myds) {
this.myds = myds;
}
Connection bind(Connection realConn) {
this.realConn = realConn;
//建立链接对象的代理类
this.proxyConn = (Connection) Proxy.newProxyInstance(this.getClass()
.getClassLoader(), new Class[] { Connection.class }, this);
return proxyConn;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
//如果是close方法
if ("close".equals(method.getName())) {
this.myds.currentCount++;
if (this.myds.currentCount < this.myds.maxCount) {
this.myds.connectionsPool.addLast(this.proxyConn);
} else {
System.out.println("hhhhhhhhhhh");
this.realConn.close();
this.myds.currentCount--;
}
}
//如果不是close方法则动态调用传入的数据库驱动的Connection对象的方法。
return method.invoke(this.realConn, args);
}
}
因此我们就需要修改数据源类MyDataSource类的创建数据库链接对象的方法createConnection方法如下,其他则不用修改
/**
* 创建链接(使用代理类的Connection)
*
* @return
* @throws SQLException
*/
private Connection createConnection() throws SQLException {
// return DriverManager.getConnection(url, user, password);
Connection realConn = DriverManager.getConnection(url, user, password);
MyConnectionHandler proxy = new MyConnectionHandler(this);
return proxy.bind(realConn);
// MyConnection myConn = new MyConnection(realConn, this);
// return myConn;
}