兴糖蹭饭笔记-基于XML&注解的SpringAOP通知

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

且挨过三冬四夏,暂受些此痛苦,雪尽后再看梅花。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值