题外话:最近有在骑车通勤,已经一个多月的时间了,晚上十点半睡,早上六点就醒,感觉还是有所变化,由最初的气喘吁吁变为呼吸均匀,速度也有所提升,也学会大撒把了。很多事情贵在坚持,每天保持一点的技术进步,日积月累就很有效果,既然选择了这条路,便只顾风雨兼程,放平心态,精益求精。(心态最重要也是最难修行的)
该篇主要讲解AOP相关的知识点,其实AOP相关的文章或博客实在是太多了,我还是想着尽量弄一个相对比较全面的,主要还是以我本身的自我理解为主,可能有些理解不够深入或者是有所偏差,也希望大家指出来,以备完善。
AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
在Spring中提供了面向切面编程的丰富支持,允许通过分离应用的业务逻辑与系统级服务(例如审计(auditing)和事务(transaction)管理)进行内聚性的开发。应用对象只实现它们应该做的——完成业务逻辑——仅此而已。它们并不负责(甚至是意识)其它的系统级关注点,例如日志或事务支持。
个人理解,AOP是通过“预编译方式”和“运行期间动态代理”实现程序功能的统一维护的一种技术或者说是一种标准。AOP是一个概念,其实现技术有AspectJ和springAOP。springAOP的实现过程中,借用了Aspectj的一些功能,包括@Aspect、@Before、@After、@PonitCut等注解,可以看到这些注解都是org.aspectj.lang.annotation下的。
说明:AspectJ设置LTW,LTW是LoadTimeWeaver, 简称LTW,LTW是AOP的一种实现方式,可以再给容器属性赋值的时候进行设置(bean生命周期),本篇不做展开,可自行了解。
当然aop远不止这些,只是提炼了些使用过程中用到更多的一些知识点,包括:概念、处理时机、实现原理(动态代理相关)及AOP的加载流程(这一块会结合源码来看)。接下来会按照这些点展开说明。
曾经有段时间,这几个概念我直接理解不了,一度觉得这都是起的什么名字啊,真费解。随着慢慢的理解深入,觉得柳暗花明又一村了。
理解概念是学习任何知识点的第一步,也是深入学习的第一步,因为不论是什么技术,如果只是到会用、功能可以实现的层面那太简单了,但是如果要知其然且知其所以然的话,概念是基础。
可切入的地方,是程序执行的一个点。例如,一个方法的执行或者一个异常的处理。在Spring AOP中,一个连接点总是代表一个方法执行。
粗暴理解就是@PointCut的地方,即为匹配连接点的表达式。在SpringAOP中,一个切入点可以只描述一个具体方法,也可以匹配多个方法。
切入点分为execution方式和annotation方式。execution可以用路径表达式指定哪些类织入切面,annotation可以指定被哪些注解修饰的代码织入切面。如下图:
其实就是处理逻辑,包括处理时机及处理操作。处理操作即为我们具体的处理逻辑,常见的包括日志、权限及异常等处理逻辑。而处理时机则指定该处理操作在什么时机进行处理执行,这儿我们说明一下处理时机相关注解概念:
@Around
设置指定的通知处理方法(该注解的value)切入到目标方法前后执行,这个功能强大比较常用。功能约等于@Before+@AfterReturning(后面会讲到)。
注意点:
@AfterReturning
设置指定的通知处理方法切入到目标方法正常执行完毕后运行。
注意点
@AfterThrowing
设置当前通知处理方法在目标方法方法运行抛出异常后执行 。
注意点
描述通知与切入点的对应关系。
就是通过动态代理,在目标对象方法中执行处理内容的过程。
Spring的AOP实现原理就是通过动态代理实现的。如果我们为Spring的某个bean配置了切面,那么Spring在创建这个bean的时候,实际上创建的是这个bean的一个代理对象,我们后续对bean中方法的调用,实际上调用的是代理类重写的代理方法。而Spring的AOP使用了两种动态代理,分别是JDK的动态代理,以及CGLib的动态代理。
Spring默认使用JDK的动态代理实现AOP,类如果实现了接口,Spring就会使用这种方式实现动态代理。JDK实现动态代理需要两个组件,首先第一个就是InvocationHandler接口。我们在使用JDK的动态代理时,需要编写一个类,去实现这个接口,然后重写invoke方法,这个方法其实就是我们提供的代理方法。然后JDK动态代理需要使用的第二个组件就是Proxy这个类,我们可以通过这个类的newProxyInstance方法,返回一个代理对象。生成的代理类实现了原来那个类的所有接口,并对接口的方法进行了代理,我们通过代理对象调用这些方法时,底层将通过反射,调用我们实现的invoke方法。
JDK的动态代理是基于反射实现。JDK通过反射,生成一个代理类,这个代理类实现了原来那个类的全部接口,并对接口中定义的所有方法进行了代理。当我们通过代理对象执行原来那个类的方法时,代理类底层会通过反射机制,回调我们实现的InvocationHandler接口的invoke方法。并且这个代理类是Proxy类的子类。
在 SpringBoot 2.x AOP中会默认使用Cglib来实现,但是Spring5中默认还是使用jdk动态代理。Spring AOP 默认使用 JDK 动态代理,如果对象没有实现接口,则使用 CGLIB 代理。当然,也可以强制使用 CGLIB 代理。
这里通过spring boot来演示下JDK动态代理,我们通过application.properties配置将代理默认设置为JDK代理。
spring.aop.proxy-target-class=false
首先需要一个接口及其实现类(放一个代码块里了):
public interface PhliInterface {
void testDynamic();
}
@Component
public class PhliServiceImpl implements PhliInterface{
@Override
public void testDynamic() {
System.out.println("目标方法执行");
}
}
其次,定义一个切面,将这个testDynamic()方法作为切入点,为它配置一个前置通知,代码如下:
@Component
@Aspect
public class TestAspect {
@Pointcut("execution(* cn.ph.software.proxy.PhliServiceImpl.*(..))")
public void pointtest() {
}
@Before(value = "pointtest()")
public void beforetest() {
System.out.println("目标方法开始之前执行");
}
}
最后,写个controller测试一下:
@RestController
@RequestMapping("/phli")
public class PhliController {
/**注意,这里只能通过PhliInterface注入,而不能直接注入PhliServiceImpl,因为在容器中,
使用JDK动态代理,Ioc容器中,存储的是一个类型为PhliInterface的代理对象*/
@Resource
private PhliInterface phliInterface;
@GetMapping("/test")
public void testPhli() {
phliInterface.testDynamic();
System.out.println(phliInterface.getClass().getName());
}
}
从结果可知,代理类的父类是Proxy,也就是说明使用的是JDK动态代理。
JDK的动态代理存在限制,那就是被代理的类必须是一个实现了接口的类,代理类需要实现相同的接口,代理接口中声明的方法。若需要代理的类没有实现接口,此时JDK的动态代理将没有办法使用,于是Spring会使用CGLib的动态代理来生成代理对象。CGLib直接操作字节码,生成类的子类,重写类的方法完成代理。
CGLib实现动态代理的原理是,底层采用了ASM字节码生成框架,直接对需要代理的类的字节码进行操作,生成这个类的一个子类,并重写了类的所有可以重写的方法,在重写的过程中,将我们定义的额外的逻辑(简单理解为Spring中的切面)织入到方法中,对方法进行了增强。而通过字节码操作生成的代理类,和我们自己编写并编译后的类没有太大区别。
跟JDK动态代理代码类似,把application.properties里面的配置去掉或者是改为true:
spring.aop.proxy-target-class=true
controller代码调整如下:
@RestController
@RequestMapping("/phli")
public class PhliController {
//直接注入PhliServiceImpl
@Resource
private PhliServiceImpl phliService;
@GetMapping("/test")
public void testPhli() {
phliService.testDynamic();
System.out.println(phliService.getClass().getName());
}
}
可以看到通知处理逻辑在目标方法之前执行,且使用的CGLib动态代理。
这儿会涉及到源码,其实看源码的过程也是学习的过程,虽然有的时候看源码也看不下去,但是也得看啊。。。
之前在动态代理的代码示例部分提到了@EnableAspectJAutoProxy注解,这个注解用于开启aop支持,但是在实例代码部分没有体现,因为在pom文件中有spring-boot-starter-web相关依赖:
org.springframework.boot
spring-boot-starter-web
2.5.0
该依赖会引入spring-boot-autoconfigure的依赖,这是自动装配的依赖,也就是会读取其下的spring.factories文件,在该文件中有下面的配置:
可以简单看下AopAutoConfiguration这个类
说明
AopAutoConfiguration等同于@EnableAspectJAutoProxy注解,该类起的作用和@EnableAspectJAutoProxy是一样的。
该类上的注解@ConditionalOnProperty,指的是在配置文件中”spring.aop.auto“的配置,如果不配置则为true。可以预料到如果将spring.aop.auto设置为false的话动态代理不会生效。
在spring.aop.auto=false基础上,在Application启动类上添加@EnableAspectJAutoProxy,再次执行:
可以看到CGLib动态代理生效。
综上,在springboot环境下,由于存在spring-boot-autoconfigure依赖,默认会注入AopAutoConfiguration配置类,该类的作用等同于@EnableAspectJAutoProxy注解,所以在这种情况下可以不加@EnableAspectJAutoProxy注解,AopAutoConfiguration可以通过spring.aop.auto属性控制。
为方便理解,当然也是为了方便自己梳理,每一步都添加相关说明。
进入registerAspectJAnnotationAutoProxyCreatorIfNecessary方法:
接下来着重看一下AnnotationAwareAspectJAutoProxyCreator
查看一下该类的类关系图:
经过排查定位查找到父类AbstractAutoProxyCreator中包含有postProcessAfterInitialization方法,该方法在bean初始化完成后执行,查看该方法注释得知,生成创建代理对象的逻辑就在这儿!!
继续跟踪查看createProxy方法,该方法最后一行:
proxyFactory.getProxy(classLoader)
SpringAOP还是有很多东西的,本篇可能没有涵盖,希望读者可以反馈下。
一些其它相关知识,可以查看我之前的博客:知识点汇总