上一篇文章《探索微服务中的权限控制:一次线上问题排查的思考》中,我们聊到面向切面编程中的切点可以在指定方法执行前后添加代码块,用于权限的控制。在切点执行时,使用了动态代理生成了对应方法的代理类。今天我们就来剖析下切点执行之前,动态代理是如何生成指定方法的代理的。
什么是动态代理?
动态代理(Dynamic Proxy)是一种在程序运行时动态创建代理对象的机制,它能够在不修改原始类代码的基础上,对目标对象的方法调用进行拦截、增强或修改等操作。
Java中动态代理有哪几种实现方式?
在 Java 中,动态代理主要有以下两种常见的实现方式,我们分别来介绍。
1. 基于Java原生反射机制的动态代理
这种方法也叫基于接口的代理,依托于 java.lang.reflect 包下的相关类和接口来实现,主要用于代理实现了接口的类,其核心是 Proxy 类和 InvocationHandler 接口。
2. 基于CGlib(Code Generation Library)的动态代理
这种方式也叫基于类(继承)的代理。依托CGLib,一个功能很强大的第三方代码生成库,它基于字节码操作技术,通过在运行时动态生成目标类的子类来实现动态代理,这种方式不仅可以代理实现了接口的类,更重要的是能够代理那些没有实现接口的普通类,其核心是Enhancer 类和MethodInterceptor 接口。
后续会专门出博客以最佳实战的方式来剖析这两者的源码
切点的动态代理使用了哪一种?
先说结论,切点的动态代理使用了第一种方式,即基于Java原生反射机制的动态代理。由上文可知,第一种动态代理的实现方式主要用于代理实现了接口的类,那么我们来看实现了JoinPoint接口的类是那个?答案是MethodInvocationProceedingJoinPoint类,即MethodInvocationProceedingJoinPoint作为被代理的类。
核心实现解析
MethodInvocationProceedingJoinPoint类主要用于在Spring AOP中基于方法调用的切面场景中,承载着方法调用相关的各种关键信息以及提供了执行目标方法和控制方法调用流程的能力。
public class MethodInvocationProceedingJoinPoint implements ProceedingJoinPoint, JoinPoint.StaticPart {
private static final ParameterNameDiscoverer parameterNameDiscoverer = new DefaultParameterNameDiscoverer(