文章目录:
1.引子
我们在写代码的时候,肯定都会不断地往代码中增加一些新的内容,也就是说在某些业务方法中增加新的功能。这样就会导致出现一些问题:👇👇👇
1. 源代码可能改动的比较多。
2. 重复代码比较多,造成一定量的代码冗余。
3. 代码难于维护。
所以在这里,就引出了Spring的另一核心技术——AOP。
2.AOP的概念
AOP(Aspect Orient Programming):面向切面编程。面向切面编程是从动态角度考虑程序运行过程。
AOP为 Aspect Oriented Programming 的缩写,意为:面向切面编程,可通过运行期动态代理实现程序功能的统一维护的一种技术。AOP 是 Spring 框架中的一个重要内容。利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
面向切面编程,就是将交叉业务逻辑封装成切面,利用 AOP 容器的功能将切面织入到主业务逻辑中。所谓交叉业务逻辑是指,通用的、与主业务逻辑无关的代码,如安全检查、事务、日志、缓存等。
若不使用 AOP,则会出现代码纠缠,即交叉业务逻辑与主业务逻辑混合在一起。这样,会使主业务逻辑变的混杂不清。例如,转账,在真正转账业务逻辑前后,需要权限控制、日志记录、加载事务、结束事务等交叉业务逻辑,而这些业务逻辑与主业务逻辑间并无直接关系。但,它们的代码量所占比重能达到总代码量的一半甚至还多。它们的存在,不仅产生了大量的“冗余”代码,还大大干扰了主业务逻辑---转账。
切面一般都是非业务功能,用来给业务方法增加新的功能,切面一般都是可以复用的。例如:日志功能、事务功能、权限检查、参数检查、统计信息等等。
上图中的日志模块、事务模块就是切面。
2.2 AOP的作用
1. 让切面功能复用。
2. 让开发人员专注业务逻辑,提高开发效率。
3. 实现业务功能和其他非业务功能的解耦合。
4. 给存在的业务方法增加功能,无需修改原来的代码。
1. Aspect:切面,给业务方法增加到功能。切面泛指交叉业务逻辑。上例中的事务处理、日志处理就可以理解为切
面。常用的切面是通知(Advice)。实际就是对主业务逻辑的一种增强。2. JoinPoint:连接点,连接切面的业务方法。连接点指可以被切面织入的具体方法。通常业务接口中的方法均为连接点。
3. Pointcut:切入点,切入点指声明的一个或多个连接点的集合。通过切入点指定一组方法。被标记为 final 的方法是不能作为连接点与切入点的。因为最终的是不能被修改的,不能被增强的。
4. Target:目标对象,目标对象指将要被增强的对象。即包含主业务逻辑的类的对象。
5. Advice:通知,通知表示切面的执行时间,Advice 也叫增强。换个角度来说,通知定义了增强代码切入到目标代码的时间点,是目标方法执行之前执行,还是之后执行等。通知类型不同,切入时间不同。
AOP中重要的三个要素:Aspect、Pointcut、Advice。意思是说:在Advice的时间、在Pointcut的位置,执行Aspect。
对于 AOP 这种编程思想,很多框架都进行了实现。Spring 就是其中之一,可以完成面向切面编程。
然而,AspectJ 也实现了 AOP 的功能,且其实现方式更为简捷,使用更为方便,而且还支持注解式开发。所以,Spring 又将 AspectJ 的对于 AOP 的实现也引入到了自己的框架中。在 Spring 中使用 AOP 开发时,一般使用 AspectJ 的实现方式。
AspectJ是一个优秀面向切面的框架,它扩展了 Java 语言,提供了强大的切面实现。
官网地址:http://www.eclipse.org/aspectj/
AspectJ框架可以使用注解和 xml配置文件这两种方式实现AOP。
3.使用AspectJ框架实现AOP的基本知识
AspectJ表示切面的执行时间,其中常用的通知有五种类型:(全部是注解,在这里被称为:通知注解)
1. @Before:前置通知。
2. @AfterReturning:后置通知。
3. @Around:环绕通知。
4. @AfterThrowing:异常通知。
5. @After:最终通知。
3.2 AspectJ的切入点表达式
AspectJ定义了专门的表达式用于指定切入点。表达式的原型是:👇👇👇
execution(modifiers-pattern? ret-type-pattern declaring-type-pattern?name-pattern(param-pattern) throws-pattern?)
切入点表达式的解释如下:👇👇👇
1. modifiers-pattern] 访问权限类型
2. ret-type-pattern 返回值类型
3. declaring-type-pattern 包名类名
4. name-pattern(param-pattern) 方法名(参数类型和参数个数)
5. throws-pattern 抛出异常类型
6. ?表示可选的部分
execution(访问权限方法返回值方法声明(参数)异常类型):红色部分是必须有的!!!
切入点表达式要匹配的对象就是目标方法的方法名。所以,execution 表达式中明显就是方法的签名。注意,表达式中黑色文字表示可省略部分,各部分间用空格分开。在其中可以使用以下符号:
举例:
1. execution(public * *(..)) :指定切入点为:任意公共方法。
2. execution(* set*(..)) :指定切入点为:任何一个以“set”开始的方法。
3. execution(* com.xyz.service.*.*(..)) :指定切入点为:定义在 service 包里的任意类的任意方法。
4. execution(* com.xyz.service..*.*(..)) :指定切入点为:定义在 service 包或者子包里的任意类的任意方法。“..”出现在类名中时,后面必须跟“*”,表示包、子包下的所有类。
5. execution(* *..service.*.*(..)) :指定所有包下的 serivce 子包下所有类(接口)中所有方法为切入点。

