要想真正了解什么是AOP你需要先了解一下一些术语
切面(Aspect)
例如我们现在使用hibernate做添加用户功能,我们先开启一个事务,然后再调用session
的save
方法进行添加,最后再提交事务.如果你忘了或者没有学过请你先看一下这篇文章.
这里的有两处不足的地方,第一处就是重复代码,比如我们现在又要添加两个功能,一个是修改 另一个是删除这样我们会不断的开启和提交事务.
第二处就是关注点,对于添加用户为例,我们只需要关注添加用户是怎么实现的就可以,而不需要关注开启事务和提交事务.
所以我们就可以提取出事务开启和提交的代码,当我们应用要执行用户添加 修改 删除的时候,再将这些代码切入到指定的方法(例如用户添加方法)中,我们将这些取出出来的代码称为切面.
连接点(Joinpoint)
连接点这个比较好理解,就是程序在运行过程中的一个点,这个点就是一个方法或者一个异常.
通知(Advice)
通知就是我们在连接点上要执行的动作,比如我们的连接点是一个方法,那么在执行这个方法前或者后来执行哪些通知(也就是一些代码).
切入点(Pointcut)
满足切入点的表达式,就是一个切入点,例如我们的切入点表达式为
* com.xxx.service.*.*(..)
上面表达式的意思是,com.xxx.service
包下的所有类所有方法,都是切入点.
通知和一个切入点表达式关联,并在满足这个切入点的连接点上运行.
织入(Weaving)
把切面织入到某个类或者对象上执行.这个动作可以是在编译时 类加载时 或运行时完成.
那么到底什么是AOP呢?
面向切面编程,在程序运行的时候,动态的织入某些代码实现程序功能的统一维护.
** AOP执行原理 **
我们为fun()
创建一个代理方法,当通过代理的``fun()方法调用原方法
fun()```时,就可以在代理方法中添加新的功能,也就是所谓的通知.
通知类型
前置通知(Before advice):在某连接点之前执行的通知,但这个通知不能阻止连接点之前的执行流程(除非它抛出一个异常)。
后置通知(After returning advice):在某连接点正常完成后执行的通知:例如,一个方法没有抛出任何异常,正常返回。
异常通知(After throwing advice):在方法抛出异常退出时执行的通知。
最终通知(After (finally) advice):当某连接点退出的时候执行的通知(不论是正常返回还是异常退出)。
环绕通知(Around Advice):包围一个连接点的通知,如方法调用。这是最强大的一种通知类型。环绕通知可以在方法调用前后完成自定义的行为。它也会选择是否继续执行连接点或直接返回它自己的返回值或抛出异常来结束执行。
下面我们看看这几个通知的创建
前置通知
public class LoggerBefore implements MethodBeforeAdvice {
@Override
public void before(Method method, Object[] objects, Object o) throws Throwable {
//method: 方法
//objects: 参数
//o: 调用类
}
}
后置通知
public class LoggerAfterReturning implements AfterReturningAdvice {
@Override
public void afterReturning(Object o, Method method, Object[] objects, Object o1) throws Throwable {
//o: 返回值
//method: 方法
//objects: 参数
//o1: 调用类
}
}
环绕通知
public class DebugInterceptor implements MethodInterceptor {
public Object invoke(MethodInvocation invocation) throws Throwable {
System.out.println("Before: invocation=[" + invocation + "]");
Object rval = invocation.proceed();
System.out.println("Invocation returned");
return rval;
}
}
异常通知
public static class CombinedThrowsAdvice implements ThrowsAdvice {
public void afterThrowing(RemoteException ex) throws Throwable {
}
public void afterThrowing(Method m, Object[] args, Object target, ServletException ex) {
}
}
这个比较特殊,方法名必须是afterThrowing
.方法的最后一个参数是必须的,前三个参数要么都写要么都不写.
那么问题来了我们怎么知道这些通知切入到哪些地方呢?
当然是我们前面提到过的切入点了,我们配置spring xml如下代码.
<aop:config>
<aop:pointcut id="bizMethods"
expression="execution(* cc.xxx.maintain.service.*.*(..))" />
<aop:advisor advice-ref="txAdvice" pointcut-ref="bizMethods" />
</aop:config>
<aop:pointcut />
标签配置切入点表达式.
<aop:advisor />
标签将通知与切入点结合起来
pointcut-ref表示使用那个切入点,advice-ref表示切入到那个类.
附上一个事务的配置
<!-- 事务管理 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<tx:advice id="txAdvice"
transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="del*" propagation="REQUIRED" />
<tx:method name="update*" propagation="REQUIRED" />
<tx:method name="*" propagation="SUPPORTS" read-only="true" />
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="bizMethods"
expression="execution(* cc.ibadboy.maintain.service.*.*(..))" />
<aop:advisor advice-ref="txAdvice" pointcut-ref="bizMethods" />
</aop:config>