您的位置:首页 > 数据库

动态代理实现数据库连接池功能(代理Connection的close方法)

2016-07-13 16:28 441 查看
package cn;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.sql.Connection;
import java.sql.DriverManager;
import java.util.LinkedList;
/**
* 在java种怎样实现动态代理呢
* 第一步,我们要有一个接口,还要有一个接口的实现类,而这个实现类呢就是我们要代理的对象,
* 所谓代理呢也就是在调用实现类的方法时,可以在方法执行前后做额外的工作,这个就是代理。
* 第二步,我们要自己写一个在要代理类的方法执行时,能够做额外工作的类,而这个类必须继承InvocationHandler接口,
* 为什么要继承它呢?因为代理类的实例在调用实现类的方法的时候,不会调真正的实现类的这个方法,
* 而是转而调用这个类的invoke方法(继承时必须实现的方法),在这个方法中你可以调用真正的实现类的这个方法。
* 第三步,在要用代理类的实例去调用实现类的方法的时候,写出下面两段代码。
*/
public class ZFXConnectionPool {

//此list种放的就就是N个连接,此连接是动态代理过的connection,修改了原来close方法
private static LinkedList<Connection> listConnection = new LinkedList<Connection>();
static{
//*****************这句话可以在根目录生成代理类的class文件*********
System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");
try {
Class.forName("com.mysql.jdbc.Driver");
for(int i = 0 ;i<3;i++){
final Connection conn = DriverManager.getConnection("jdbc:mysql:///mysql", "root", "root");
Object obj = Proxy.newProxyInstance(ZFXConnectionPool.class.getClassLoader(), new Class[]{Connection.class},
new InvocationHandler() {
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
if("close".equals(method.getName())){
synchronized (listConnection) {
listConnection.add((Connection) proxy);
listConnection.notifyAll();//唤醒
return null;
}
}else{
//注意:第一个参数不能填 proxy , 因为proxy 是代理后的对象,会死循环执行invoke方法
return method.invoke(conn, args);
}
}
});
listConnection.add((Connection) obj);
}

} catch (Exception e) {
e.printStackTrace();
}
}

public static Connection getConn(){
synchronized (listConnection) {
if(listConnection.size() == 0){
try {
listConnection.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
return getConn();
}else{
return listConnection.removeFirst();

}
}
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: