AOP(面向切面编程):将横切关注点与业务逻辑相分离
范例:日志、声明式事务、安全、缓存。
描述切面的常用术语有通知(advice)、切点(pointcut)和连接点(join point)
1.通知(Advice):切面的工作
Spring切面可以应用5种类型的通知:
- 前置通知(Before):在目标方法被调用之前调用通知功能;
- 后置通知(After):在目标方法完成之后调用通知,此时不会关心方法的输出是什么;
- 返回通知(After-returning):在目标方法成功执行之后调用通知;
- 异常通知(After-throwing):在目标方法抛出异常后调用通知;
- 环绕通知(Around):通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行自定义的行为。
2.连接点(Join point)
连接点是在应用执行过程中能够插入切面的一个点,这个点可以是调用方法时、抛出异常时、甚至修改一个字段时。
3.切点(Poincut)
切点的定义会匹配通知所要织入的一个或多个连接点。(在哪些连接点)
4.切面(Aspect)
切面是通知和切点的结合——它是什么,在何时和何处完成其功能。
5.引入(Introducution)
引入允许我们向现有的类添加新方法或属性。
6.织入(Weaving)
织入是把切面应用到目标对象并创建新的代理对象的过程。
在目标对象的生命周期里有多个点可以进行织入:
- 编译期:切面在目标类编译时被织入。这种方式需要特殊的编译器。AspectJ的织入编译器就是以这种方式织入切面的。
- 类加载期:切面在目标类加载到JVM时被织入。这种方式需要特殊的类加载器(ClassLoader),它可以在目标类被引入应用之前增强该目标类的字节码。AspectJ 5的加载时织入(load-timeweaving,LTW)就支持以这种方式织入切面。
- 运行期:切面在应用运行的某个时刻被织入。一般情况下,在织入切面时,AOP容器会为目标对象动态地创建一个代理对象。Spring AOP就是以这种方式织入切面的。
Spring提供了4种类型的AOP支持:
- 基于代理的经典Spring AOP;(曾经很nb)
- 纯POJO切面;(需要XML配置)
- @AspectJ注解驱动的切面; (本质依然是Spring基于代理的AOP)
- 注入式AspectJ切面(适用于Spring各版本)。
在Spring AOP中,要使用AspectJ的切点表达式语言来定义切点。
execution指示器是我们在编写切点定义时最主要使用的指示器。在此基础上,使用其他指示器来限制所匹配的切点。
eg:
假设我们需要配置的切点仅匹配concert包。在此场景下,可以使用within()指示器来限制匹配:
eg:
在这里,我们希望在执行Performance的perform()方法时应用通知,但限定bean的ID为woodstoc
同理:
在某些场景下,限定切点为指定的bean或许很有意义,但我们还可以用非操作为除了特定ID以外的其他bean应用通知。
使用注解来创建切面是AspectJ 5所引入的关键特性。
eg:从演出的角度来看,观众是非常重要的,但是对演出本身的功能来讲,它并不是核心,这是一个单独的关注点。因此,将观众定义为一个切面,并将其应用到演出上就是较为明智的做法。
@AspectJ注解表明Audience不仅仅是一个POJO,还是一个切面。
@Pointcut注解能够在一个@AspectJ切面内定义可重用的切点。
eg:
目前还不能转换为切面的代理,需要如下操作:
如果使用JavaConfig,可以在配置类的类级别上通过使用EnableAspectJ-AutoProxy注解启用自动代理功能:
假如在Spring中要使用XML来装配bean的话,那么需要使用Spring aop命名空间中的
----------------------------------------------------------------------------------------------------------
关于环绕通知(@Around):
环绕通知是最为强大的通知类型。它能够让你所编写的逻辑将被通知的目标方法完全包装起来。实际上就像在一个通知方法中同时编写前置通知和后置通知。
当要将控制权交给被通知的方法时,它需要调用ProceedingJoinPoint的proceed()方法。
-----------------------------------------------------------------------------------------------------------------------------
eg:
拿出定义切点的表达式:
切面可以为Spring bean添加新方法。原理:当引入接口的方法被调用时,代理会把此调用委
托给实现了新接口的某个其他对象。
借助于AOP的引入功能,我们可以不必在设计上妥协或者侵入性地改变现有的实现:
通过@DeclareParents注解,将Encoreable接口引入到Performance bean中。
@DeclareParents注解由三部分组成:
- value属性指定了哪种类型的bean要引入该接口。在本例中,也就是所有实现Performance的类型。(标记符后面的加号表示是Performance的所有子类型,而不是Performance本身。)
- defaultImpl属性指定了为引入功能提供实现的类。在这里,我们指定的是DefaultEncoreable提供实现。
- @DeclareParents注解所标注的静态属性指明了要引入了接口。在这里,我们所引入的是Encoreable接口。
基于注解的配置要优于基于Java的配置,基于Java的配置要优于基于XML的配置。但是,如果你需要声明切面,但是又不能为通知类添加注解的时候(没有源码,或者不想将AspectJ注解放到你的代码之中),那么就必须转向XML配置了。
eg1:在xml中配置前置通知和后置通知
同理,也可以使用@Pointcut注解消除了这些重复的内容:
eg2:在xml中配置环绕通知
eg3:在xml中 通知带参数怎么办
如果你将这个切点表达式与上一个程序清单4.6中的表达式进行对比会发现它们几乎是相同的。唯一的差别在于这里使用and关键字而不是“&&”(因为在XML中,“&”符号会被解析为实体的开始)
eg4:过切面引入新的功能(使用Spring aop命名空间中的
SpringBoot 有四个核心:
- 自动配置:针对很多Spring应用程序常见的应用功能,Spring Boot能自动提供相关配置。
- 起步依赖:告诉Spring Boot需要什么功能,它就能引入需要的库。 (特殊的Maven依
赖和Gradle依赖)- 命令行界面:这是Spring Boot的可选特性,借此只需写代码就能完成完整的应用程序,无需传统项目构建。
- Actuator:能够深入 运行中的Spring Boot应用程序,一探究竟。
其他几个部分旨在简化Spring开发,而Actuator则要提供在运行时检视应用程序内部情况的能力。包括如下细节:
- Spring应用程序上下文里配置的Bean
- Spring Boot的自动配置做的决策
- 应用程序取到的环境变量、系统属性、配置属性和命令行参数
- 应用程序里线程的当前状态
- 应用程序最近处理过的HTTP请求的追踪情况
- 各种和内存用量、垃圾回收、Web请求以及数据源用量相关的指标