切入点表达式(注解和xml配置两种形式)
作用:通过表达式定位一个或多个具体的连接点
语法:execution([权限修饰符] [返回值类型] [简单类名/全类名] [方法名]([参数列表]))
上一段代码:这是基于注解形式的AOP。方法中的注解看后面(通知)。
package com.thekingqj.spring;
import java.util.Arrays;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;
@Component
@Aspect
public class CalcAspect { //切面
@After("execution(public int com.thekingqj.spring.CalcDaoImpl.add(int,int))") //切入点表达式
public void clacAfter(JoinPoint jp) { //通知
System.out.println("后置通知");
}
@Before("execution(* *.*(..))")
public void calcBefore(JoinPoint jp) {
System.out.println("前置通知:名称"+jp.getSignature().getName()+"参数"+Arrays.toString(jp.getArgs()));
}
@AfterReturning(value="execution(* *.*(..))",returning="result")//切入点对象受影响的位置
public Object calcAfterReturn(JoinPoint jp,Object result) {//JoinPoint连接点 获取数据(方法名称,参数)
System.out.println("返回通知"+result);
return result;
}
@AfterThrowing(value="execution(* *.*(..))", throwing="e")
public void calcThrow(JoinPoint jp,Exception e) {
System.out.println("异常通知");
System.out.println(e);
}
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd">
<!--扫描该包下的所有组件进行装配-->
<context:component-scan base-package="com.thekingqj.spring"></context:component-scan>
<!--支持注解AOP-->
<aop:aspectj-autoproxy ></aop:aspectj-autoproxy>
</beans>
至于、AOP前面已经说了,是一种动态代理的模式,这个CalcAspect就是切面,即非核心业务中的其他业务生成的类。当主方法中调用核心业务时,通过注解就可以封装核心业务返回一个代理对象。
基于XML配置的AOP:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd">
<bean id="calc" class="com.thekingqj.spring.xml.CalcDaoImpl"></bean>
<bean id="aspect" class="com.thekingqj.spring.xml.CalcAspect"></bean>
<aop:config >
<aop:pointcut expression="execution(* *.*(..))" id="cut" /><!--切入点表达式-->
<aop:aspect ref="aspect" ><!--切面-->
<aop:after method="clacAfter" pointcut-ref="cut" />
<aop:before method="calcBefore" pointcut-ref="cut" />
</aop:aspect>
</aop:config>
</beans>
通知:通知就是具体的方法
- 在具体的连接点上要执行的操作。
- 一个切面可以包括一个或者多个通知。
通知所使用的注解的值往往是切入点表达式
前置通知、后置通知、环绕通知、返回通知、异常通知
前置通知:
- 在方法执行之前执行的通知
- 使用@Before注解
后置通知:
- 后置通知是在连接点完成之后执行的,即连接点返回结果或者抛出异常的时候
- 使用@After注解
环绕通知:
- 环绕通知是所有通知类型中功能最为强大的,能够全面地控制连接点,甚至可以控制是否执行连接点。
- 对于环绕通知来说,连接点的参数类型必须是ProceedingJoinPoint。它是 JoinPoint的子接口,允许控制何时执行,是否执行连接点。
- 在环绕通知中需要明确调用ProceedingJoinPoint的proceed()方法来执行被代理的方法。如果忘记这样做就会导致通知被执行了,但目标方法没有被执行。
- 注意:环绕通知的方法需要返回目标方法执行之后的结果,即调用 joinPoint.proceed();的返回值,否则会出现空指针异常。
返回通知:
- 返回通知:无论连接点是正常返回还是抛出异常,后置通知都会执行。如果只想在连接点返回的时候记录日志,应使用返回通知代替后置通知。
- 使用@AfterReturning注解,在返回通知中访问连接点的返回值
①在返回通知中,只要将returning属性添加到@AfterReturning注解中,就可以访问连接点的返回值。该属性的值即为用来传入返回值的参数名称
②必须在通知方法的签名中添加一个同名参数。在运行时Spring AOP会通过这个参数传递返回值
③原始的切点表达式需要出现在pointcut属性中
异常通知:
- 异常通知:只在连接点抛出异常时才执行异常通知
- 将throwing属性添加到@AfterThrowing注解中,也可以访问连接点抛出的异常。Throwable是所有错误和异常类的顶级父类,所以在异常通知方法可以捕获到任何错误和异常。
- 如果只对某种特殊的异常类型感兴趣,可以将参数声明为其他异常的参数类型。然后通知就只在抛出这个类型及其子类的异常时才被执行