SpringAOP中各种增强方法

Spring配置文件


<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.3.xsd
		http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd">

	<context:component-scan base-package="com.zzu">context:component-scan>
	<aop:aspectj-autoproxy proxy-target-class="false">aop:aspectj-autoproxy>
beans>
//待增强的目标类
import org.springframework.stereotype.Service;

@Service
public class CalculatorService implements ICalculatorService {

	@Override
	public int mul(int a, int b) {
		int result = a*b;
		return result;
	}
}

//Test类
import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.zzu.calculator.ICalculatorService;

public class Test {

	public static void main(String[] args) {
		ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("beans.xml");
		ICalculatorService calculator = applicationContext.getBean(ICalculatorService.class);
		int result = calculator.mul(1, 2);
		System.out.println(result);
		applicationContext.close();
	}
}

1、Before
前置增强(又称前置通知):在目标方法执行之前执行

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class CalculatorAspect {

	@Before("execution(public int com.zzu.calculator.CalculatorService.mul(int, int))")
	public void before(JoinPoint jp) {
		Object obj = jp.getTarget();
		Object[] args = jp.getArgs();
		System.out.println(obj.getClass().getName()+":The "+methodName+" method begins.");
		System.out.println(obj.getClass().getName()+":Parameters of the "+methodName+" method: ["+args[0]+","+args[1]+"]");
	}
}

执行Test类结果:

com.zzu.calculator.CalculatorService:The mul method begins.
com.zzu.calculator.CalculatorService:Parameters of the mul method: [1,2]
2

2、After
保留Before,查看整体效果。

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class CalculatorAspect {

	//前置增强
	@Before("execution(public int com.zzu.calculator.CalculatorService.mul(int, int))")
	public void before(JoinPoint jp) {
		Object obj = jp.getTarget();
		Object[] args = jp.getArgs();
		String methodName = jp.getSignature().getName();
		System.out.println(obj.getClass().getName()+":The "+methodName+" method begins.");
		System.out.println(obj.getClass().getName()+":Parameters of the "+methodName+" method: ["+args[0]+","+args[1]+"]");
	}
	//后置增强
	@After("execution(public int com.zzu.calculator.CalculatorService.mul(int,int))")
	public void after(JoinPoint jp) {
		Object obj = jp.getTarget();
		String methodName = jp.getSignature().getName();
		System.out.println(obj.getClass().getName()+":The "+methodName+" method ends.");
	}
}

执行Test类结果:

com.zzu.calculator.CalculatorService:The mul method begins.
com.zzu.calculator.CalculatorService:Parameters of the mul method: [1,2]
com.zzu.calculator.CalculatorService:The mul method ends.
2

后置增强(又称后置通知):在目标方法执行后执行,无论目标方法运行期间是否出现异常。
例如我们在目标类的方法中添加异常:

if(true) {
	throw new ArithmeticException("两数之积不能为0");
}

SpringAOP中各种增强方法_第1张图片
出现异常,但是并没有影响后置增强的输出。
注意:后置增强无法获取目标方法执行结果,可以在返回增强中获取。
3、AfterReturning
返回增强(又称返回通知):在目标方法正常结束后执行,可以获取目标方法的执行结果。
保留前置增强和后置增强,查看整体效果

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class CalculatorAspect {

	//前置增强
	@Before("execution(public int com.zzu.calculator.CalculatorService.mul(int, int))")
	public void before(JoinPoint jp) {
		Object obj = jp.getTarget();
		Object[] args = jp.getArgs();
		String methodName = jp.getSignature().getName();
		System.out.println(obj.getClass().getName()+":The "+methodName+" method begins.");
		System.out.println(obj.getClass().getName()+":Parameters of the "+methodName+" method: ["+args[0]+","+args[1]+"]");
	}
	//返回增强
	@AfterReturning(value="execution(public int com.zzu.calculator.CalculatorService.mul(int,int))",returning="result")
	public void afterReturning(JoinPoint jp,Object result) {
		Object obj = jp.getTarget();
		String methodName = jp.getSignature().getName();
		System.out.println(obj.getClass().getName()+":Result of the "+methodName+" method:"+result);
	}
	//后置增强
	@After("execution(public int com.zzu.calculator.CalculatorService.mul(int,int))")
	public void after(JoinPoint jp) {
		Object obj = jp.getTarget();
		String methodName = jp.getSignature().getName();
		
		System.out.println(obj.getClass().getName()+":The "+methodName+" method ends.");
	}
}

执行Test类结果:

com.zzu.calculator.CalculatorService:The mul method begins.
com.zzu.calculator.CalculatorService:Parameters of the mul method: [1,2]
com.zzu.calculator.CalculatorService:The mul method ends.
com.zzu.calculator.CalculatorService:Result of the mul method:2
2

发现上面的最后两条信息与我们期望顺序相反,这与After和AfterReturning的执行顺序有关,一般来说更改不了。
4、AfterThrowing
异常增强(又称异常通知):目标方法抛出异常之后执行,可以访问到异常对象,且可以指定在出现哪种异常时才执行增强代码

package com.zzu.calculator;

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 CalculatorAspect {

	//前置增强
	@Before("execution(public int com.zzu.calculator.CalculatorService.mul(int, int))")
	public void before(JoinPoint jp) {
		Object obj = jp.getTarget();
		Object[] args = jp.getArgs();
		String methodName = jp.getSignature().getName();
		System.out.println(obj.getClass().getName()+":The "+methodName+" method begins.");
		System.out.println(obj.getClass().getName()+":Parameters of the "+methodName+" method: ["+args[0]+","+args[1]+"]");
	}
	//返回增强
	@AfterReturning(value="execution(public int com.zzu.calculator.CalculatorService.mul(int,int))",returning="result")
	public void afterReturning(JoinPoint jp,Object result) {
		Object obj = jp.getTarget();
		String methodName = jp.getSignature().getName();
		System.out.println(obj.getClass().getName()+":Result of the "+methodName+" method:"+result);
	}
	//异常增强
	@AfterThrowing(value="execution(public int com.zzu.calculator.CalculatorService.mul(int,int))",throwing="exception")
	public void afterThrowing(JoinPoint jp,NullPointerException exception) {
		Object obj = jp.getTarget();
		String methodName = jp.getSignature().getName();
		System.out.println(obj.getClass().getName()+":Exception of the method "+methodName+": "+exception.getMessage());
	}
	//后置增强
	@After("execution(public int com.zzu.calculator.CalculatorService.mul(int,int))")
	public void after(JoinPoint jp) {
		Object obj = jp.getTarget();
		String methodName = jp.getSignature().getName();
		
		System.out.println(obj.getClass().getName()+":The "+methodName+" method ends.");
	}
}

在目标方法中添加代码:

if(true) {
	throw new ArithmeticException("两数之积不能为0");
}

执行Test类结果:

com.zzu.calculator.CalculatorService:The mul method begins.
com.zzu.calculator.CalculatorService:Parameters of the mul method: [1,2]
com.zzu.calculator.CalculatorService:The mul method ends.
Exception in thread "main" java.lang.ArithmeticException: 两数之积不能为0

发现没有结果输出,也就是说异常出现时目标方法中的result没有返回,同时返回增强没有执行。
到此我们得出执行顺序:前置增强–>后置增强–>返回增强或异常增强
5、Around
Around是环绕增强,此注解可以完成以上所有注解的整合:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class CalculatorAspect {

	@Around("execution(public int com.zzu.calculator.CalculatorService.mul(int,int))")
	public Object result(ProceedingJoinPoint jp) {//环绕增强实现以上所有增强,
		Object result = null;
		Object obj = jp.getTarget();
		Object[] args = jp.getArgs();
		String methodName = jp.getSignature().getName();
		try {
			try {
				//前置增强
				System.out.println(obj.getClass().getName()+":The mul method begins.");
				System.out.println(obj.getClass().getName()+":Parameters of the mul method: ["+args[0]+","+args[1]+"]");
				result = jp.proceed();//调用目标对象方法
			} finally {
				//后置增强
				System.out.println(obj.getClass().getName()+":The mul method ends.");
			}
			//返回增强
			System.out.println(obj.getClass().getName()+":Result of the "+methodName+" method:"+result);
		} catch (Throwable e) {
			//异常增强
			System.out.println(obj.getClass().getName()+":Exception of the method "+methodName+": "+e.getMessage());
		}
		return result;
	}
}

执行Test类结果:

com.zzu.calculator.CalculatorService:The mul method begins.
com.zzu.calculator.CalculatorService:Parameters of the mul method: [1,2]
Exception in thread "main" org.springframework.aop.AopInvocationException: Null return value from advice does not match primitive return type for: public abstract int com.zzu.calculator.ICalculatorService.mul(int,int)
	at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:227)
	at com.sun.proxy.$Proxy8.mul(Unknown Source)
	at com.zzu.test.Test.main(Test.java:12)
com.zzu.calculator.CalculatorService:The mul method ends.
com.zzu.calculator.CalculatorService:Exception of the method mul: 两数之积不能为0

你可能感兴趣的:(框架)