前面几天介绍了Spring,IOC容器,bean的生命周期和初始化及销毁过程,注解的使用。
今天介绍一节纯干活,AOP的实现原理,看完这一篇,面试过程中,再也不要和面试官尬聊:面向切面编程!引用场景在日志,方法增强等等方面!
本文通过对AOP是什么,使用场景,实现原理等方面,结合代码等方式图文介绍,用最简短的语言文字描述,彻底掌握AOP的使用方式,面试中介绍AOP时跟面试官侃侃而谈,面试官只会对你说一个字:牛逼!
一、AOP介绍
AOP:面向切面编程【底层就是动态代理】,指程序在运行期间动态的将某段代码切入到指定方法位置进行的编程方式。
使用AOP有什么好处?
场景:例如定义一个方法,方法中要计算两者相除,方法中需要验证被除数的数据是否符合逻辑(比如不能为0),我们当然可以通过 if 条件方式将逻辑处理好。但是代码略显臃肿,此时可以通过引入AOP,在执行计算之前将判断逻辑处理好,以及在运算之后将对应的后置相关逻辑也加上,这样不仅不需要在代码中写大量的 if 等逻辑代码,而且通过这种方式,对方法进行统一封装引用。
接下来进行实例演示
建立配置类
在pom.xml中引入spring-aspects依赖
新建实体类Calculator.java
在Calculator的div()方法运行之前,记录一下日志,运行后也记录一下日志,如果运行出异常,也将异常打印一下。可能实现的方式就会在方法体中每一个都写上相应的代码,这样方法多了,会显得非常的冗余及耦合。
此时需要一个日志切面类,来将比如日志等方法进行统一管理。
日志切面类的方法需要动态感知到div()方法运行到哪里了,然后再执行,如果除法开始,就日志开始方法,也叫通知方法,分以下几种:
前置通知:logStart(),在目标方法(div)运行之前运行 (@Before)
后置通知:logEnd(),在目标方法(div)运行结束之后运行,无论正常或异常结束 (@After)
返回通知:logReturn,在目标方法(div)正常返回之后运行 (@AfterReturning)
异常通知:logException,在目标方法(div)出现异常后运行(@AfterThrowing)
环绕通知:以上没写,动态代理,手动执行目标方法运行joinPoint.procced(),最底层通知,手动指定执行目标方法(@Around),执行之前相当于前置通知,执行之后相当于返回通知,其实就是通过反射执行目标对象的连接点处的方法。
给日志切面类LogAspect的方法标注何时运行(即通知注解)
怎么加入呢?
Spring无法区别【切面类】,哪个是业务类,在这里给LogAspects类加上@Aspect注解即可声明为切面类。
同时,若不想区分切入了哪个方法及参数类型和个数,可以有如下指定方式:
在这里又发现问题:注解里的内容是冗余重复的, 公共的代码应该抽出来封装
有了以上操作, 我们还需要将切面类和被切面的类, 都加入到容器中,在配置类中引入
是不是就完了呢?并没有
需要开启基于注解的AOP模式
给配置类中加@EnableAspectJAutoProxy[一定得加上,关键]
注意:在spring以后会有很多@EnableXXXX,表示开启某项功能,取代XML配置
测试一下:新建一个测试类Cap10Test.java
同学们在测试的过程中,应该怎么测?很容易出问题。
大家可能会这么写:
没用到容器,肯定是不行的,获取bean时使用IOC容器取出bean
总结:
AOP看起来很麻烦, 只要3步就可以了:
1. 将业务逻辑组件和切面类都加入到容器中, 告诉spring哪个是切面类(@Aspect)
2. 在切面类上的每个通知方法上标注通知注解, 告诉Spring何时运行(写好切入点表达式,参照官方文档)
3. 开启基于注解的AOP模式 @EableXXXX
小伙伴们可以看到,无论是在工作中,还是面试回答中,对AOP仅仅停留在面向切面编程层面上是远远不够的,还需要深入对里面的流程进行了解,本文针对AOP的原理,实现方式进行了介绍,希望后续小伙伴在工作和面试中通过本文AOP的介绍,对AOP的引用能够更加灵活,面试回答更加游刃有余。
(感谢来自享学IT老师们的分享)
下一篇将深入到AOP源码,对AOP进行深入解析。
如果觉得好的话,可以关注,点赞和转发呀~
关注微信公众号【Java极客思维】
Java极客思维
微信扫一扫,关注公众号