[TOC]
**动态代理有以下特点:**
1.代理对象,不需要实现接口
2.代理对象的生成,是利用JDK的API,动态的在内存中构建代理对象(需要我们指定创建代理对象/目标对象实现的接口的类型)
3.动态代理也叫做:**JDK代理,接口代理**
**JDK中生成代理对象的API**
代理类所在包:java.lang.reflect.Proxy
JDK实现代理只需要使用newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:
~~~
static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )
~~~
注意该方法是在Proxy类中是静态方法,且接收的三个参数依次为:
* `ClassLoader loader,`:指定当前目标对象使用类加载器,获取加载器的方法是固定的
* `Class<?>[] interfaces,`:目标对象实现的接口的类型,使用泛型方式确认类型
* `InvocationHandler h`:事件处理,执行目标对象的方法时,会触发事件处理器的方法,会把当前执行目标对象的方法作为参数传入
### 代码示例:
接口类IUserDao.java以及接口实现类,目标对象UserDao是一样的,没有做修改.在这个基础上,增加一个代理工厂类 (ProxyFactory.java),将代理类写在这个地方,然后在测试类(需要使用到代理的代码)中先建立目标对象和代理对象的联系,然后代用代理 对象的中同名方法
### 代理工厂类:
ProxyFactory.java
~~~
package com.yc.model.ProxyFactory;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.security.PublicKey;
/*
创建动态代理对象
动态代理不需要实现,但是要指定接口类型
*/
public class ProxyFactory {
private Object target;
public ProxyFactory(Object target){
this.target=target;
}
public Object getProxyInstance(){
return Proxy.newProxyInstance(target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("开始事务2");
//运用反射执行目标对象方法
Object returnValue = method.invoke(target, args);
System.out.println("提交事务2");
return returnValue;
}
});
}
}
~~~
### 测试类
~~~
package com.yc.model;
import com.yc.model.ProxyFactory.ProxyFactory;
import com.yc.model.Service.IUserDao;
import com.yc.model.Service.Impl.UserDao;
public class Test {
public static void main (String[] args){
//目标对象
IUserDao target =new UserDao();
//代理对象
IUserDao proxy = (IUserDao)new ProxyFactory(target).getProxyInstance();
//执行的是代理的方法
proxy.save();
}
}
~~~
### 结果:
![](https://img.kancloud.cn/03/9c/039c32cefce51d36c62e6649303817c9_1034x452.png)