AOP术语
概念 | 说明 |
---|---|
切入点(Pointcut) | 切点指明哪些方法调用上嵌入横切逻辑。 |
连接点(Joinpoint) | 连接点是在应用执行过程中能够插入切面的一个点。 |
切面(Aspect) | 切面是通知和切点的结合。通知和切点共同定义了切面的全部内容——它是什么,在何时和在何处完成其功能。 |
通知(Advice) | 切面的工作被称为通知。通知定义了切面是什么以及何时使用。除了描述切面要完成的工作,通知还解决了何时执行这个工作的问题。 |
通知类型 | 说明 |
---|---|
前置通知(Before) | 在目标方法被调用之前调用通知功能 |
后置通知(After) | 在目标方法完成之后调用通知 |
返回通知(After-returning) | 在目标方法成功执行之后调用通知 |
异常通知(After-throwing) | 在目标方法抛出异常后调用通知 |
环绕通知(Around) | 通知包裹了被通知的方法,在被通知的方法调用之前和之后执行自定义的行为 |
基于XML配置的SpringAOP开发
测试类:
//接口类
public interface BankService {
public void transfer(String source, String target, BigDecimal money);
}
//接口实现类
public class XTBankService implements BankService{
public String transfer(String source, String target, BigDecimal money) {
System.out.println(source + "向" + target + "转账:" + money + "元。");
return "OLD RETURNVAL!";
}
}
public class Main {
public static void main(String[] args) {
ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext-xml-anno.xml");
BankService bsFromIoC = ioc.getBean("bankService",BankService.class);
System.out.println(bsFromIoC.transfer("A", "B", new BigDecimal("999")));
}
}
XML配置:
<!--配置所需的Bean-->
<bean id="bankService"
class="com.zzxtit.spring.aop.xml.XTBankService"></bean>
<bean id="loggerAspect"
class="com.zzxtit.spring.aop.xml.LoggerAspect"></bean>
<!--配置切入点与切面-->
<aop:config>
<aop:pointcut expression="execution(* com.zzxtit.spring.aop.xml.*.*(..))" id="loggerPointCut"/>
<aop:aspect ref="loggerAspect">
<!--配置通知属性-->
<aop:before method="loggerBefore" pointcut-ref="loggerPointCut"/>
<aop:after method="loggerAfter" pointcut-ref="loggerPointCut"/>
<aop:after-returning method="loggerAfterReturn" pointcut-ref="loggerPointCut" returning="returnVal"/>
<aop:around method="loggerAround" pointcut-ref="loggerPointCut"/>
</aop:aspect>
</aop:config>
单个通知运行时实况
前置通知(Before)
public void loggerBefore(JoinPoint jp) {
System.out.println("----------------前置通知----------------");
Object[] args = jp.getArgs();
for(Object arg:args) {
System.out.println("arg:" + arg);
}
}
<!--XML配置-->
<aop:before method="loggerBefore" pointcut-ref="loggerPointCut"/>
//运行结果
----------------前置通知----------------
arg:A
arg:B
arg:1999.9
A向B转账:1999.9元。
后置通知(Before)
public void loggerAfter(JoinPoint jp) {
System.out.println("----------------后置通知----------------");
}
<!--XML配置-->
<aop:after method="loggerAfter" pointcut-ref="loggerPointCut"/>
//运行结果
A向B转账:1999.9元。
----------------后置通知----------------
返回通知(After-returning)
//注意:Object returnVal对象名需与XML中returning属性值相同
public Object loggerAfterReturn(JoinPoint jp, Object returnVal) {
System.out.println("----------------返回通知----------------");
System.out.println("returnVal=====>" + returnVal);
return "returnVal=====>" + returnVal;
}
<!--XML配置-->
<aop:after-returning method="loggerAfterReturn" pointcut-ref="loggerPointCut" returning="returnVal"/>
//运行结果
A向B转账:1999.9元。
----------------返回通知----------------
returnVal=====>OLD RETURNVAL!
异常通知(After-throwing)
public class XTBankService implements BankService{
public String transfer(String source, String target, BigDecimal money) {
//修改目标对象使其抛出异常触发异常通知
System.out.println(source + "向" + target + "转账:" + money + "元。");
throw new RuntimeException("故意出的异常!");
// return "OLD RETURNVAL!";
}
}
public void loggerAfterThrowing(JoinPoint jp, Exception exception) {
System.out.println("----------------异常通知----------------");
System.out.println("exception);
}
<!--XML配置-->
<aop:after-throwing method="loggerAfterThrowing" pointcut-ref="loggerPointCut" throwing="exception"/>
//运行结果
A向B转账:1999.9元。
----------------异常通知----------------
java.lang.RuntimeException: 故意出的异常!
Exception in thread "main" java.lang.RuntimeException: 故意出的异常!
环绕通知(Around)
public Object loggerAround(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("----------------环绕通知----------------");
Object returnVal = pjp.proceed();
System.out.println("OLD returnVal=====>" + returnVal);
returnVal = "NEW RETURNVAL!";//修改函数返回值
System.out.println("NEW returnVal=====>" + returnVal);
return returnVal;
}
<!--XML配置-->
<aop:around method="loggerAround" pointcut-ref="loggerPointCut"/>
//运行结果
----------------环绕通知----------------
A向B转账:1999.9元。
OLD returnVal=====>OLD RETURNVAL!
NEW returnVal=====>NEW RETURNVAL!
多个通知同时运行时的次序(除异常通知)
----------------前置通知----------------
arg:A
arg:B
arg:1999.9
----------------环绕通知----------------
A向B转账:1999.9元。
OLD returnVal=====>OLD RETURNVAL!
NEW returnVal=====>NEW RETURNVAL!
----------------返回通知----------------
returnVal=====>NEW RETURNVAL!
----------------后置通知----------------
基于注解的SpringAOP开发
测试类:
@Service("bankService")
public class XTBankService implements BankService{
//代码同上 此处省略
}
public class Main {
public static void main(String[] args) {
ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext-xml-anno.xml");
//获取ioc容器中的bankService
BankService bsFromIoC = ioc.getBean("bankService",BankService.class);
bsFromIoC.transfer("A", "B", new BigDecimal("999"));
}
}
XML配置:
<!-- 配置Bean的自动扫描器 -->
<context:component-scan base-package="com.zzxtit.spring.aop.anno"></context:component-scan>
<!-- 启动AspectJ 注解支持 -->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
注解编写
通知类Java类编写:使用@Aspect 和@Component标记为切面的Spring Bean组件。
//硬编码设置切入点,当只有单属性时value可以省略
@Before("execution(* com.zzxtit.spring.aop.anno.*.*(..))")
public void loggerBefore(JoinPoint jp) {
//代码同上 此处省略
}
@After(value = "execution(* com.zzxtit.spring.aop.anno.*.*(..))")
public void loggerAfter(JoinPoint jp) {
//代码同上 此处省略
}
@AfterReturning(value = "execution(* com.zzxtit.spring.aop.anno.*.*(..))",returning = "returnVal")
public Object loggerAfterReturn(JoinPoint jp, Object returnVal) {
//代码同上 此处省略
}
//需要复用切入点时,可以先配置切入点函数后调用
@Pointcut("execution(* com.zzxtit.spring.aop.anno.*.*(..))")
public void pointCut() {
}
@Before("pointCut()")
public void loggerBefore(JoinPoint jp) {
//代码同上 此处省略
}
当对同一方法设置多个切面时,切入点存在既定运行顺序。若要控制其顺序需要使用@Order(数字)标记,@Order()内数字越小,运行优先级越高。
@Order(1)
@Aspect
@Component
public class AuthAspect {
@Pointcut("execution(* com.zzxtit.spring.aop.anno.*.*(..))")
public void pointCut() {
}
@Before("pointCut()")
public void loggerBefore(JoinPoint jp) {
System.out.println("----------权限--前置通知----------------");
}
@After(value = "execution(* com.zzxtit.spring.aop.anno.*.*(..))")
public void loggerAfter(JoinPoint jp) {
System.out.println("----------权限--后置通知----------------");
}
}
@Order(0)
@Aspect
@Component
public class LoggerAspect {
//代码同上 此处省略
}
//运行结果
----------------环绕通知----------------
----------------前置通知----------------
----------权限--前置通知----------------
A向B转账:1999.9
----------权限--后置通知----------------
OLD returnVal=====>OLD RETURNVAL!
NEW returnVal=====>NEW RETURNVAL!
----------------后置通知----------------
----------------返回通知----------------
returnVal=====>NEW RETURNVAL!
Summary
且挨过三冬四夏,暂受些此痛苦,雪尽后再看梅花。