动态代理
# JDK动态代理和CGLIB动态代理
# 1. 什么是代理模式
代理模式(Proxy Pattern)给某一个对象提供一个代理,并用代理对象控制原对象的引用。代理对象再客户端和目标对象之间起到中介作用。
代理模式是常用的结构型设计模式之一,当直接访问某些对象存在问题时可以通过一个代理对象来间接访问,为了保证客户端使用的透明性,所访问的真实对象与代理对象需要实现相同的接口。代理模式属于结构型设计模式,属于GOF23设计模式
代理模式可以分为静态代理和动态代理两种类型,而动态代理中又分为 JDK 动态代理和 CGLIB 代理两种。

代理模式包含如下角色:
- Subject(抽象主体角色)抽象主体角色声明了真实主体和代理主体的共同接口,这样依赖在任何使用真实主体的地方都可以使用代理主体。客户端需要针对抽象主体角色进行编程。
- Proxy(代理主体角色)代理主体角色内部包含对真实主体的引用,从而可以在任何时候操作真是主体对象。在代理主体角色中提供一个与真实主体角色相同的接口,以便在任何时候都可以代替真实主体。代理主体角色还可以控制对真实主体的使用,负责在需要的时候创建和删除真是主体对象,并对真实主体对象的使用加以约束。代理角色通常在客户端调用所引用的真实主体操作之前或之后还需要执行其他操作,而不仅仅是单纯的调用真是主体对象中的操作。
- RealSubject(真是主体角色)真是主体角色定义了代理角色所代表的真实对象,在真实主体角色中实现了真实的业务操作,客户端可以通过代理主体角色间接调用真是主体角色重定义的方法。
# 2. 代理模式的优点
- 代理模式能将代理对象与真实被调用的目标对象分离。
- 一定程度上降低了系统柜的耦合度,扩展性好。
- 可以起到保护目标对象的作用。
- 可以对目标对象的功能增强
# 3. 代理模式的缺点
- 代理模式会造成系统设计中类的数量增加。
- 在客户端和目标对象增加一个代理对象,会造成请求处理速度变慢。
# 4. JDK动态代理
在java的动态代理机制中,有两个重要的类或接口,一个是 InvocationHandler(Interface)、另外一个则是Prox(Class),这个类和接口是实现我们动态代理所必须用到的。
# 4.1 InvocationHandler
每一个动态代理类都必须要实现 InvocationHandler 这个接口,并且每个代理类的实例都关联了一个handler,当我们通过代理对象调用一个方法的时候,这个方法的调用就会被转发为由 InvocationHandler 这个接口的 invoke 方法来进行调用。
InvocationHandler 这个接口的唯一一个方法 invoke 方法:
Object invoke(Object proxy, Method method, Object[] args) throws Throwable
这个方法一共接受三个参数,那么这三个参数分别代表如下:
- proxy :指代 JDK 动态生成的最终代理对象。
- method :指代的是我们所要调用真是对象的某个方法的 Method 对象。
- args : 指代的是调用真实对象某个方法时接受的参数。
# 4.2 Proxy
Proxy 这个类的作用就是来动态创建一个代理对象的类,它提供了许多的方法,但是我们用的最多的就是 newProxyInstance 这个方法:
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler handler) throws IllegalArgumentException
这个方法的作用就是得到一个动态的代理对象,其接收三个参数,我们来看看这三个参数所代表的含义:
- loader :ClassLoader对象,定义了由那些 ClassLoader 来对生成的代理对象进行加载,即代理类的类加载器。
- interfaces : Interface 对象的数组,表示的是我将要给我需要代理的对象提供一组什么接口,如果我提供了一组接口给它,那么这个代理对象就宣称实现了该接口(多态),这样我就能调用这组接口中的方法了。
- Handler :InvocationHandler 对象,表示的是当我这个动态代理对象在调用方法的时候,会关联到哪一个 InvocationHandler 对象上。
所以我们所说的 DynamicProxy (动态代理类)是这样一种class : 它是在运行时生成的 class ,在生成它时你必须提供一组 interface 给它,然后改 class 就宣称它实现了这些 interface。这个 DynamicProxy 其实就是一个 Proxy,它不会做实质性的工作,在生成它的实例时你必须提供一个 handler,由它接管实际的工作。
# 4.3 JDK 动态代理实例
- 创建接口类
public interface HelloInterface {
void sayHello();
}
2
3
- 创建被代理类,实现接口
/**
* 被代理类
*/
public class HelloImpl implements HelloInterface{
@Override
public void sayHello() {
System.out.println("hello");
}
}
2
3
4
5
6
7
8
9
- 创建InvocationHandler实现类
/**
* 每次生成动态代理类对象时都需要指定一个实现了InvocationHandler接口的调用处理器对象
*/
public class ProxyHandler implements InvocationHandler{
private Object subject; // 这个就是我们要代理的真实对象,也就是真正执行业务逻辑的类
public ProxyHandler(Object subject) {// 通过构造方法传入这个被代理对象
this.subject = subject;
}
/**
*当代理对象调用真实对象的方法时,其会自动的跳转到代理对象关联的handler对象的invoke方法来进行调用
*/
@Override
public Object invoke(Object obj, Method method, Object[] objs)
throws Throwable {
Object result = null;
System.out.println("可以在调用实际方法前做一些事情");
System.out.println("当前调用的方法是" + method.getName());
result = method.invoke(subject, objs);// 需要指定被代理对象和传入参数
System.out.println(method.getName() + "方法的返回值是" + result);
System.out.println("可以在调用实际方法后做一些事情");
System.out.println("------------------------");
return result;// 返回method方法执行后的返回值
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
- 测试
public class Mytest {
public static void main(String[] args) {
//第一步:创建被代理对象
HelloImpl hello = new HelloImpl();
//第二步:创建handler,传入真实对象
ProxyHandler handler = new ProxyHandler(hello);
//第三步:创建代理对象,传入类加载器、接口、handler
HelloInterface helloProxy = (HelloInterface) Proxy.newProxyInstance(
HelloInterface.class.getClassLoader(),
new Class[]{HelloInterface.class}, handler);
//第四步:调用方法
helloProxy.sayHello();
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
- 结果
可以在调用实际方法前做一些事情
当前调用的方法是sayHello
hello
sayHello方法的返回值是null
可以在调用实际方法后做一些事情
------------------------
2
3
4
5
6
# 4.4 JDK 动态代理步骤
JDK 动态代理分为以下几步:
- 拿到被代理对象的引用,并且通过反射获取到它的所有的接口。
- 通过 JDK Proxy 类重新生成一个新的类,同时新的类要实现被代理类所实现的所有的接口。
- 动态生成 Java 代码,把新加的业务逻辑方法由一定的逻辑代码去调用。
- 编译新生成的 Java 代码 .class。
- 将新生成的Class文件重新加载到JVM中运行。
所以说 JDK 动态代理的核心是通过重写被代理对象所实现的接口中的方法重新生成代理类来实现的,那么加入被代理对象没有实现接口呢?那么这时候就需要 CGLIB 动态代理了。
# 5 CGLIB 动态代理
JDK 动态代理是通过重写被代理对象实现的接口中的方法来实现,而CGLIB是通过集成部诶代理对象来实现和 JDK 动态代理需要实现指定接口一样,CGLIB 也要求代理对象必须要实现 MethodInterceptor 接口,并重写其唯一的方法 intercept 。
CGLib 采用了非常底层的字节码技术,其原理是通过字节码技术为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势植入横切逻辑。(利用ASM开源包,对代理对象类的 class 文件加载进来,通过修改其字节码生成子类来处理)。
注意 :因为CGLIB 是通过集成目标类来重写其方法来实现的,故而如果是 final 和 private 方法则无法被重写,也就无法被代理。
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib-nodep</artifactId>
<version>2.2</version>
</dependency>
2
3
4
5
# 5.1 CGLIB 核心类
net.sf.cglib.proxy.Enhancer :主要增强类,通过字节码技术动态创建委托类的子类实例;
Enhancer 可能是 CGLIB 中最常用的一个类,和Java1.3 动态代理中引入的Proxy类差不多。和Proxy不同的是,Enhancer 技能够代理普通的 class,也能够代理接口。Enhancer 创建一个被代理对象的子类并且拦截所有的方法调用(包括从 Object 中继承的 toString 和 hashCode 方法)。Enhancer 不能够拦截 final 方法,例如 Object.getClass() 方法,这是由于 Java final 方法语义决定的。基于同样的道理,Enhancer 也不能对 final 类进行 代理操作。这也是 Hibernate 为什么不能持久化 final class 的原因。
net.sf.cglib.proxy.MethodInterceptor :常用的方法拦截器接口,需要实现 intercept 方法,实现具体拦截处理:
public java.lang.Object intercept(java.lang.Object obj,
java.lang.reflect.Method method,
java.lang.Object[] args,
MethodProxy proxy) throws java.lang.Throwable{}
2
3
4
5
obj : 动态生成的代理对象。
method : 实际调用的方法。
args : 调用方法入参。
net.sf.cglib.proxy.MethodProxy :java Method类的代理类,可以实现委托类对象的方法的调用;常用方法:methodProxy.invokeSuper(proxy, args);在拦截方法内可以调用多次。
# 5.2 CGLIB 代理实例
- 创建被代理类
public class SayHello {
public void say() {
System.out.println("hello")
}
}
2
3
4
5
- 创建代理类
/**
*代理类
*/
public class ProxyCglib implements MethodInterceptor{
private Enhancer enhancer = new Enhancer();
public Object getProxy(Class clazz){
//设置需要创建子类的类
enhancer.setSuperclass(clazz);
enhancer.setCallback(this);
//通过字节码技术动态创建子类实例
return enhancer.create();
}
//实现MethodInterceptor接口方法
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("可以在调用实际方法前做一些事情");
//通过代理类调用父类中的方法
Object result = proxy.invokeSuper(obj, args);
System.out.println("可以在调用实际方法后做一些事情");
return result;
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
- 测试
public class Mytest {
public static void main(String[] args) {
ProxyCglib proxy = new ProxyCglib();
//通过生成子类的方式创建代理类
SayHello proxyImp = (SayHello)proxy.getProxy(SayHello.class);
proxyImp.say();
}
}
2
3
4
5
6
7
8
9
- 结果
可以在调用实际方法前做一些事情
hello
可以在调用实际方法后做一些事情
2
3
# 5.3 CGLIB 动态代理实现分析
CGLIB 动态代理采用了 FastClass 机制,其分别为代理类和被代理类各生成一个FastClass,这个 FastClass 类会为代理类或被代理类的方法分配一个 index(int类型)。这个index 当作一个入参,FastClass 就可以直接定位要调用的方法直接进行调用,这样省去了反射调用,所以调用效率比 JDK 动态代理通过反射调用更高。
但是我们看上面的源码也可以明显看到,JDK 动态代理只生成一个文件,而 CGLIB 生成了三个文件,所以生成代理对象的过程会更复杂。
# 6 JDK 和 CGLIB 动态代理对比
JDK 动态代理是实现了呗代理对象啊所实现的接口,CGLIB 是继承了被代理对象。JDK 和 CGLIB 都是在运行期生成字节码,JDK 是直接写 Class 字节码,CGLib 使用 ASM 框架写 Class 字节码,Cglib 代理实现更为复杂,生成代理类的效率比 JDK 代理低。
JDK 调用代理方法,是通过反射机制调用,CGLIB 是通过 FastClass 机制直接调用方法,CGLIB 执行效率更高。
# 6.1 原理区别:
java 动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用 InvocationHandler 来处理。核心是实现 InvocationHandler 接口,使用 invoke() 方法进行面向切面的处理,调用相应的通知。
- 如果目标对象实现了接口,默认情况下会采用 JDK 的动态代理实现 AOP。
- 如果目标对象实现了接口,可以强制使用 CGLIB 实现 AOP。
- 如果目标对象没有实现了接口,必须采用 CGLIB 库,spring 会自动在 JDK 动态代理和 CGLIB 之间转换
# 6.2 性能区别:
CGLib底层采用ASM字节码生成框架,使用字节码技术生成代理类,在jdk6之前比使用Java反射效率要高。唯一需要注意的是,CGLib不能对声明为final的方法进行代理,因为CGLib原理是动态生成被代理类的子类。
在jdk6、jdk7、jdk8逐步对JDK动态代理优化之后,在调用次数较少的情况下,JDK代理效率高于CGLIB代理效率,只有当进行大量调用的时候,jdk6和jdk7比CGLIB代理效率低一点,但是到jdk8的时候,jdk代理效率高于CGLIB代理。
# 6.3 各自局限
JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理。
cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
| 类型 | 机制 | 回调方式 | 适用场景 | 效率 |
|---|---|---|---|---|
| JDK动态代理 | 委托机制,代理类和目标类都实现了同样的接口,InvocationHandler持有目标类,代理类委托InvocationHandler去调用目标类的原始方法 | 反射 | 目标类是接口类 | 效率瓶颈在反射调用稍慢 |
| CGLIB动态代理 | 继承机制,代理类继承了目标类并重写了目标方法,通过回调函数MethodInterceptor调用父类方法执行原始逻辑 | 通过FastClass方法索引调用 | 非接口类、非final类,非final方法 | 第一次调用因为要生成多个Class对象,比JDK方式慢。多次调用因为有方法索引比反射快,如果方法过多,switch case过多其效率还需测试 |
# 7 静态代理和动态代理的区别
静态代理只能通过手动完成代理操作,如果被代理类增加新的方法,代理类需要同步新增,违背开闭原则。
动态代理采用在运行时动态生成代码的方式,取消了对被代理类的扩展限制,遵循开闭原则。
若动态代理要对目标类的增强逻辑扩展,结合策略模式,只需要新增策略类便可完成,无需修改代理类的代码。
← 单元测试 01 【数据库概述】→