4、Java 动态代理
2021-08-01 本文已影响0人
我要离开浪浪山
1、静态代理
代理模式给某一个对象提供一个代理对象,并由代理对象控制对原对象的引用。通俗的来讲代理模式就是我们生活
中常见的中介。比如你按照小卡片上的电话打过去寻求服务,一般不是由本人,可能是一个成年雄性接听电话,然
而真正做事情的可能是另一个小姐姐。
目的:
(1)通过引入代理对象的方式来间接访问目标对象,防止直接访问目标对象给系统带来的不必要复杂性;
(2)通过代理对象对访问进行控制;
代理模式一般会有三个角色:

抽象角色:指代理角色和真实角色对外提供的公共方法,一般为一个接口
真实角色:需要实现抽象角色接口,定义了真实角色所要实现的业务逻辑,以便供代理角色调用。也就是真正的业
务逻辑在此。
代理角色:需要实现抽象角色接口,是真实角色的代理,通过真实角色的业务逻辑方法来实现抽象方法,并可以附
加自己的操作。将统一的流程控制都放到代理角色中处理!
- 静态代理在使用时,需要定义接口或者父类,被代理对象与代理对象一起实现相同的接口或者是继承相同父类。一般
来说,被代理对象和代理对象是一对一的关系,当然一个代理对象对应多个被代理对象也是可以的。- 静态代理,一对一则会出现时静态代理对象量多、代码量大,从而导致代码复杂,可维护性差的问题,一对多则代
理对象会出现扩展能力差的问题。
2、动态代理
在运行时再创建代理类和其实例,因此显然效率更低。要完成这个场景,需要在运行期动态创建一个Class。JDK提
供了 Proxy 来完成这件事情。
基本使用如下:
//抽象角色
interface Api {
void test(String a);
}
//真实角色
class ApiImpl{
@Override
public void test(String a) {
System.out.println("真实实现:" + a);
}
}
//创建真实角色实例
ApiImpl api = new ApiImpl();
//JDK动态代理: Proxy.newProxyInstance(getClass().getClassLoader(),
new Class[]{Api.class}, //JDK实现只能代理接口
new InvocationHandler() {
@Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//执行真实对象方法
return method.invoke(api, args);
}
});
实际上, Proxy.newProxyInstance 会创建一个Class,与静态代理不同,这个Class不是由具体的.java源文件编译
而来,即没有真正的文件,只是在内存中按照Class格式生成了一个Class。
String name = Api.class.getName()+"$Proxy0";
//生成代理指定接口的Class数据 byte[] bytes = ProxyGenerator.generateProxyClass(name, new Class[]{Api.class}); FileOutputStream fos = new FileOutputStream("lib/" + name+".class"); fos.write(bytes); fos.close();
然后可以在生成的文件中查看我们的代理类:

在初始化时,获得 method 备用。而这个代理类中所有方法的实现变为:

这里的 h 其实就是 InvocationHandler 接口,所以我们在使用动态代理时,传递的 InvocationHandler 就是一个
监听,在代理对象上执行方法,都会由这个监听回调出来。