转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/6776247.html
一:AOP基础概念
(1)通知(增强)Advice
通知,其实就是我们从众多类中提取出来的重复功能代码。也即是我们切入到某个类的某个方法点处要执行的操作。如:权限验证、日志操作等。
(2)连接点 Join point
连接点是针对程序运行的类来说的,指:一个类在运行过程中,有哪些点是可以被切入,执行通知操作进行加工的。
(3)切点 Pointcut
切点是针对连接点的子集。切点 往往与 通知 协同生效,切点是 通知选择切入的连接点 的合集,切点一般用正则表达式来指定。
(4)切面 Aspect
切面是通知和切点的容器。
(5)引入 Introduction
引入允许我们在不影响原有类的情况下,向现有的类添加新方法或属性,从而无需修改这些现有类的情况下,让他们具有新的行为和状态。
既然是新增了,又怎么做到不影响原有类呢?我们用 强加一个父类 的方法来实现。
指定类A,为它强加一个父类 类B,类B中定义了新增的一系列方法。由于现在A继承B,所以B中的方法就可以被A调用了。因此,getBean("A")获得A对象a后,就可以用a.fun_of_B()来调用类B中新增的方法了。
(6)织入 Weaving
织入是指切面在指定的切点处连接到目标对象中,从而创建出一个被通知(增强)过的对象,一般有:编译时织入、类加载时织入、运行时织入。你可以这样理解:一个类A在上面三个织入时期其中一个,经过织入过程,得到了一个加工过的 类A' 。类A'与类A的不同在于,切点处方法的前、后增加了一些操作(那就是通知)。
二:Spring中的AOP实现
AOP有两种实现,分别是预编译实现、运行时实现。SpringAOP底层使用的是运行时实现,但是对预编译实现也有支持。至于什么是预编译实现、运行时实现,放在Java原理深入的博文中进行探究,此处仅学习SpringAOP的使用。
1:定义切面bean类
切面是通知的容器,里面存放了同一模块的通知。例如:日志切面类,里面包含了info、error、warning等不同级别的日志输出方法。
2:注册切面bean
在applicationcontext.xml配置文件中配置这个bean:
<bean id="myaspect" class="路径">
。。。
</bean>
3:配置切面
1)切面的配置有着严格的层次,外层是<aop:config>标签,指明这是AOP配置,然后是<aop:aspect>标签,说明这是一个切面。
<bean id="aspectBean" class="路径">//注册切面bean
。。。
</bean>
<aop:config>
<aop:aspect id="myAspect" ref="aspectBean">//配置一个切面,依赖切面bean
。。。
</aop:aspect>
</aop:config>
2) 在切面标签下,配置这个切面拥有的切入点。切入点是用正则表达式(用 * 代表路径名或方法名,..代表参数列表)来说明的,也可以用完整的路径进行精确的指定。
SpringAOP中只用到了方法作为切入点。更加精细的切入可以自行用运行时动态代理来实现,如:方法内某语句作为切入点。
<bean id="aspectBean" class="路径">
。。。
</bean>
<aop:config>
<aop:aspect id="myAspect" ref="aspectBean">
<aop:pointcut expression="execution(返回值 路径.方法名(..))" id="pointcut_1"/>//配置一个切入点,参数列表为所有情况都匹配
<aop:pointcut expression="execution(返回值 路径.方法名(Type1,Type2)) and args(name1,name2)" id="pointcut_1"/>//精确配置一个切入点,参数列表处为参数类型,args为参数名称 。。。
</aop:aspect>
</aop:config>
3):配置具体通知的起效位置(即把切入点配置给具体的通知)
<bean id="aspectBean" class="路径">
。。。
</bean>
<aop:config>
<aop:aspect id="myAspect" ref="aspectBean">
<aop:pointcut expression="execution(返回值 路径.方法名(参数列表))" id="pointcut_1"/>//配置一个切入点 <aop:通知类型 method="切面bean类中的一个通知名" pointcut-ref="point_id" />//法一:用已经定义的切入点来配置
<aop:通知类型 method="切面bean类中的一个通知名" pointcut="execution(返回值 路径.方法名(参数列表))" />//法二:自定义一个只用一次的切入点来配置 。。。
</aop:aspect>
</aop:config>
通知类型指定了被配置的通知方法在切入点起效的时机:切入点前?后?或者前后都执行?还是抛出异常后才执行?主要有以下几种:
(1)aop:around :自定义在切入点方法执行前、后的操作,这个通知方法的定义有一定的格式:参数列表第一个参数必须是切入点
public Object around_advice(ProceedingJoinPoint point,......)//把切入点传进来,其他参数照常传
{
Object res=null;
try{
自定义切入点执行前的操作;
point.proceed();//执行切入点方法
自定义切入点执行后的操作;
}catch(Throwable e){
....
}
return res;
}
(2)aop:before:JointPoint前执行通知
(3)aop:after-returning:JointPoint方法执行完毕,正常返回值并退出方法后,执行该通知
(4)aop:after:切入点方法代码执行完最后一行后,执行该通知。它与after-returning不同,它无论切入点是否正常退出,都会执行。所以也叫finally advice。
(5)aop:after-throwing:JointPoint抛出异常时执行。注意:切入点方法抛出异常后,则不是正常返回退出,所以此时after-returning通知是不会执行的。这种通知可以通过throwing属性指定抛出什么异常时执行通知。
(6)aop:introduction:JointPoint调用完毕后
4)配置Introductions
SpringAOP中的引入是通过“新增父类”的形式来实现的,所用标签是 <aop:declare-parents>。考虑到扩展性,这里还用到了面向接口编程的原则,为匹配的类新增的父类是某个接口的实现类。也就是说:引入,不仅为匹配类新增了父类,还新增了被实现的接口,因此可以用接口来接收getBean的对象,然后调用新增的方法。
首先:定义要引入的父类接口以及接口实现类
package com.ygj; interface Person{//父类接口:定义了要引入通知对象的新增方法
void driveCar();
} class Father implements Person{//接口实现类:实现了新增方法
public void driveCar(){
System.out.println("开车去玩");
}
}
然后,在切面中进行配置,通过正则表达式匹配要增强的目标类,为它们定义上述父类接口以及实现类:
<aop:config>
<aop:aspect id="myAspect" ref="aspectBean">
<aop:declare-parents
types-matching="要增强的类的路径,可用正则表达式匹配多个类,如:com.people.Son"
implement-interface="引入的父类接口完整路径,如:com.ygj.Person"
default-impl="接口的实现类,如:com.ygj.Father"/>
。。。
</aop:aspect>
</aop:config>
最后,可以用引入的父接口去获取被增强的类对象,并且可以调用引入的方法了:
Person person=(Person)applicationContext.getBean("Son");//面向接口编程
person.driveCar();//调用引入的方法
5)Advisors
与事务管理advice配合使用,见下一篇博文。