1 AOP
AOP:Aspect Oriented Programming(面向切面编程),它可以通过预编译或运行期动态代理的方式实现程序功能的 统一维护 的一种技术,比如:日志记录、性能统计、安全控制、事务处理(如数据库的连接,关闭)、异常处理等等。
AOP 的实现方式:
- 预编译:AspectJ
- 运行期动态代理(JDK动态代理(有接口)、CGLib动态代理(无接口)):SpringAOP、JbossAOP
1.1几个基础的概念
- 切面(aspect):一个关注点的 模块化,这个关注点可能横切多个对象。(个人理解就是通知的集合对象实例)
- 连接点(JoinPoint):程序执行过程的某个特定的点
- 切入点(PointCut):匹配连接点的断言,在AOP中通知和切入点关联
- 通知(advice):在切面的某个特定的连接点上执行的动作
- 引入(Introduction):在不修改类代码的前提下,为类添加新的方法和属性(添加父类)
- 目标对象(target object):被一个或多个切面所通知的对象
- AOP代理(AOP Proxy):AOP 框架创建的对象,用来实现切面契约(aspect context),包括通知方法执行的功能
- 织入(weaving):把切面连接到其它的应用程序类型或对象上,并创建一个被通知对象
1.1.1 advice(通知)的类型
advice(通知)的类型:
- 前置通知(Before advice):在某个连接点(JoinPoint)之前执行的通知,但不能阻止连接点的执行,除非它抛出一个异常
- 返回后通知(After advice):在某个连接点 正常完成后 执行的通知
- 抛出异常后通知(After throwing advice):在某个连接点抛出异常后退出时执行的通知
- 后通知(After advice/Finally advice):在某个连接点退出时执行的通知(无论是正常返回还是异常返回)
- 环绕通知(Around advice):包围一个连接点的通知
1.1.2 Spring 的 AOP
Spring 的 AOP 是纯 java 的实现,无需特殊的编译过程,目前只支持方法执行的连接点,因为它只侧重于提供一种AOP的实现用以和IOC容器之间的整合。
2 基于配置文件 aop实现(schema base AOP)
在基于XML配置文件 aop 的实现中,Spring 的所有的切面和通知器都必须放置在 <aop:config>
标签中,一个配置文件可以包含多个 <aop:config>
标签,每个标签里可以包含 <aop:aspect>
、 <aop:poincut>
和 advisor
标签。
2.1 <aop:poincut>
标签与 poincut
属性
<aop:poincut>
标签的 expression
属性有很多形式,比如下图,不需要死记硬背,更多的需要自身查找
当不使用 <aop:poincut>
标签,可以直接在 advice
标签中使用 poincut
属性,此时的属性值就是对应的 expression
属性值。
expression
属性赋值可以精确到方法的参数,此时在末尾添加 and args(argName)
就可以在通知中使用到该参数,使用poincut
属性也同理,如下:
<aop:around method="aroundInit" poincut="execution(* cn.seiei.bean.test(String,int)) and args(testStr, testInt)" />
<!-- 注意括号的位置,以及 * 号后有空格 -->
2.2 advice 标签
advice 标签有 <aop:before>
、<aop:after-returning>
、<aop:after-throwing>
、<aop:after>
、<aop:round>
。
它们共同的都有 method
,poincut-ref
属性,其中 method
就是 <aop:aspect>
指定的 Bean 中的方法名字,poincut-ref
属性是指对应的 <aop:poincut>
切入点声明的 ID
2.2.1 <aop:round>
标签
<aop:round>
标签指定的方法的 第一个参数必须是 ProceedingJoinPoint
类型,这个 ProceedingJoinPoint
类型的参数有一个 proceed
方法返回 Object
类型,这表示具体业务的代码的执行,其返回值是具体业务的代码执行的返回值,此时需要把这个返回值在声明的 method
方法中进行返回,在 proceed
方法的执行前后可以添加其它代码用以到达使用 before
或 after
的功能。
2.3 <aop:declare-parents>
标签
<aop:declare-parents>
标签的作用是在不修改类代码的前提下,为类添加新的方法和属性,它实质就是创建一个接口和实现该接口的类,然后让匹配的类继承于它,所以能让原来的类转化为新创建的父类,从而获得新方法。
<aop:declare-parents>
标签有三个属性:types-matching
,implement-interface
以及 defualt-impl
。
types-matching
表示需要指定,匹配到的类;implement-interface
表示接口的声明defualt-impl
表示实现这个接口的类
3 基于 AspectJ 的形式配置
@AspecetJ
的风格类似纯 Java 注解,Spring可以使用 AspectJ 的形式做切入点的解析,当AOP运行的时候还是纯的String AOP,对 AspectJ 的编译器或织入没有依赖性。
@Aspect
注解:表示切面,@Aspect
注解不能被 Spring 以类路径的形式扫描识别,所以需要添加@component
注解将它注册到 IOC 容器中@PoinCut(execution(...))
注解:表示切入点,注解的方法返回值必须为void
@Before(execution(...))/@before(注解了切入点的方法名())
@AfterRunning(execution(...))/@AfterRunning(注解了切入点的方法名())
@After(execution(...))/@After(注解了切入点的方法名())
@AfterThrowing(execution(...))/@AfterThrowing(注解了切入点的方法名())
@Around(execution(...))/@Around(注解了切入点的方法名())