JoinPoint和ProceedingJoinPoint对象
-
JoinPoint
对象封装了SpringAop中切面方法的信息,在切面方法中添加JoinPoint参数,就可以获取到封装了该方法信息的JoinPoint对象. -
ProceedingJoinPoint
对象是JoinPoint的子接口,该对象只用在@Around的切面方法中
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | @Aspect @Component public class aopAspect { /** * 定义一个切入点表达式,用来确定哪些类需要代理 * execution(* aopdemo.*.*(..))代表aopdemo包下所有类的所有方法都会被代理 */ @Pointcut ( "execution(* aopdemo.*.*(..))" ) public void declareJoinPointerExpression() {} /** * 前置方法,在目标方法执行前执行 * @param joinPoint 封装了代理方法信息的对象,若用不到则可以忽略不写 */ @Before ( "declareJoinPointerExpression()" ) public void beforeMethod(JoinPoint joinPoint){ System.out.println( "目标方法名为:" + joinPoint.getSignature().getName()); System.out.println( "目标方法所属类的简单类名:" + joinPoint.getSignature().getDeclaringType().getSimpleName()); System.out.println( "目标方法所属类的类名:" + joinPoint.getSignature().getDeclaringTypeName()); System.out.println( "目标方法声明类型:" + Modifier.toString(joinPoint.getSignature().getModifiers())); //获取传入目标方法的参数 Object[] args = joinPoint.getArgs(); for ( int i = 0 ; i |
切点表达式
- 在Spring AOP中,连接点始终代表方法的执行。切入点是与连接点匹配的,切入点表达语言是以编程方式描述切入点的方式。
- 切入点(Poincut)是定义了在“什么地方”进行切入,哪些连接点会得到通知。显然,切点一定是连接点
- 切点是通过
@Pointcut
注解和切点表达式
定义的。@Pointcut注解可以在一个切面内定义可重用
的切点。
execute表达式
*
代表匹配任意修饰符及任意返回值,参数列表中..
匹配任意数量的参数
可以使用&&、||、!、三种运算符来组合切点表达式,表示与或非的关系
- 1.拦截任意公共方法
execution(public * *(..))
- 2.拦截以set开头的任意方法
execution(* set*(..))
- 3.拦截类或者接口中的方法
1 2 | 拦截AccountService(类、接口)中定义的所有方法 execution(* com.xyz.service.AccountService.*(..)) |
4.拦截包中定义的方法,不包含子包中的方法
1 2 | 拦截com.xyz.service包中所有类中任意方法,**不包含**子包中的类 execution(* com.xyz.service.*.*(..)) |
5.拦截包或者子包中定义的方法
1 2 | 拦截com.xyz.service包或者子包中定义的所有方法 execution(* com.xyz.service..*.*(..)) |
通知分类
@Before
- 前置通知: 在方法执行之前执行
- 前置通知使用
@Before
注解 将切入点表达式值作为注解的值
@After
- 后置通知, 在方法执行之后执行
- 后置通知使用
@After
注解 ,在后置通知中,不能访问目标方法执行的结果
@AfterRunning
- 返回通知, 在方法返回结果之后执行
- 返回通知使用
@AfterRunning
注解
@AfterThrowing
- 异常通知, 在方法抛出异常之后执行
- 异常通知使用
@AfterThrowing
注解
@Around
- 环绕通知, 围绕着方法执行
- 环绕通知使用
@Around
注解
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 | package com.jason.spring.aop.impl; import java.util.Arrays; import java.util.List; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.AfterReturning; import org.aspectj.lang.annotation.AfterThrowing; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.springframework.stereotype.Component; //把这个类声明为一个切面 //1.需要将该类放入到IOC 容器中 @Component //2.再声明为一个切面 @Aspect public class LoggingAspect { //声明该方法是一个前置通知:在目标方法开始之前执行 哪些类,哪些方法 //作用:@before 当调用目标方法,而目标方法与注解声明的方法相匹配的时候,aop框架会自动的为那个方法所在的类生成一个代理对象,在目标方法执行之前,执行注解的方法 //支持通配符 //@Before("execution(public int com.jason.spring.aop.impl.ArithmeticCaculatorImpl.*(int, int))") @Before ( "execution(* com.jason.spring.aop.impl.*.*(int, int))" ) public void beforeMethod(JoinPoint joinPoint){ String methodName = joinPoint.getSignature().getName(); List<object data-origwidth= "" data-origheight= "" style= "width: 1264px;" > args = Arrays.asList(joinPoint.getArgs()); System.out.println( "The method " + methodName + " begins " + args); } /** * @Description: 在方法执行后执行的代码,无论该方法是否出现异常 * @param joinPoint */ @After ( "execution(* com.jason.spring.aop.impl.*.*(int, int))" ) public void afterMethod(JoinPoint joinPoint){ String methodName = joinPoint.getSignature().getName(); List<object data-origwidth= "" data-origheight= "" style= "width: 789px;" > args = Arrays.asList(joinPoint.getArgs()); System.out.println( "The method " + methodName + " end " + args); } /** * * @Description: 在方法正常结束后执行代码,放回通知是可以访问到方法的返回值 * * @param joinPoint */ @AfterReturning ( value= "execution(* com.jason.spring.aop.impl.*.*(..))" , returning= "result" ) public void afterReturning(JoinPoint joinPoint ,Object result){ String methodName = joinPoint.getSignature().getName(); System.out.println( "The method " + methodName + " end with " + result); } /** * * @Description: 在目标方法出现异常时会执行代码,可以访问到异常对象,且,可以指定出现特定异常时执行通知代码 * * @param joinPoint * @param ex */ @AfterThrowing (value= "execution(* com.jason.spring.aop.impl.*.*(..))" ,throwing= "ex" ) public void afterThrowting(JoinPoint joinPoint, Exception ex){ String methodName = joinPoint.getSignature().getName(); System.out.println( "The method " + methodName + " occurs exceptions " + ex); } /** * * @Description: 环绕通知需要携带 ProceedingJoinPoint 类型的参数 * 环绕通知 类似于 动态代理的全过程 * ProceedingJoinPoint:可以决定是否执行目标方法 * 环绕通知必须有返回值,返回值即为目标方法的返回值 * * @param proceedingJoinPoint */ @Around ( "execution(* com.jason.spring.aop.impl.*.*(..))" ) public Object around(ProceedingJoinPoint proceedingJoinPoint){ Object result = null ; String methodName = proceedingJoinPoint.getSignature().getName(); //执行目标方法 try { //前置通知 System.out.println( "The method " + methodName + "begin with" + Arrays.asList(proceedingJoinPoint.getArgs())); result = proceedingJoinPoint.proceed(); //后置通知 System.out.println( "The method " + methodName + "end with" + result); } catch (Throwable e) { //异常通知 System.out.println( "The method occurs exception : " + e); throw new RuntimeException(); } //后置通知 System.out.println( "The method " + methodName + "end with" + result); return result; } } </object></object> |
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持IT俱乐部。