ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
[TOC] # 简介 # 准备类 ~~~ public class UserServiceImpl { public Object save(String name) { System.out.println("保存用户---"); return 1; } } ~~~ # 定义通知 准备的类还是用UserServiceImpl ~~~ package aspect; 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.aspectj.lang.annotation.Pointcut; //通知类 //表示该类是一个通知类 //order指定优先级,值越小优先级越大 @Order(100) @Aspect public class MyAdvice { /** * 声明切点(对那些方法拦截) */ @Pointcut("execution(* com.spring.service.*ServiceImpl.*(..))") public void pc() { System.out.println("pc方法"); } /** * 前置通知 */ @Before("MyAdvice.pc()") public void before(JoinPoint joinPoint) { System.out.println("---"); //使用前置通知可以完成日志记录,权限控制 System.out.println("拦截的目标类: " + joinPoint.getSignature().getDeclaringTypeName()); System.out.println("拦截的方法名称: "+ joinPoint.getSignature().getName()); System.out.println("输入的参数: " + Arrays.asList(joinPoint.getArgs())); System.out.println("这是前置通知"); } /** * 后置通知(如果出现异常不会调用) */ @Before("MyAdvice.pc()") public void before(JoinPoint joinPoint) { System.out.println("---"); //使用前置通知可以完成日志记录,权限控制 System.out.println("拦截的目标类: " + joinPoint.getSignature().getDeclaringTypeName()); System.out.println("拦截的方法名称: "+ joinPoint.getSignature().getName()); System.out.println("输入的参数: " + Arrays.asList(joinPoint.getArgs())); System.out.println("这是前置通知"); } /** * 后置通知(如果出现异常不会调用) */ @AfterReturning(value = "execution(* com.spring.service.*ServiceImpl.*(..))", returning = "val") public void afterReturning(JoinPoint joinPoint, Object val) { /** * 第二个参数可以获取目标方法的返回值 * 注意:需要在配置文件中配置 returning="val" */ System.out.println("---"); System.out.println("返回值: " + val); System.out.println("这是后置通知(如果出现异常不会调用)"); } /** * 环绕通知 */ @Around("execution(* com.spring.service.*ServiceImpl.*(..))") public Object around(ProceedingJoinPoint pjp) throws Throwable { //获取方法名 String methodName = pjp.getSignature().getName(); //获取参数 Object[] args = pjp.getArgs(); Object result = null; try { //前置通知 System.out.println("这是环绕通知之前的方法: " + methodName + " 开始,参数: " + Arrays.asList(args)); //执行方法 result = pjp.proceed(); //返回通知 System.out.println("方法: " + methodName + " 返回 " + result); } catch (Throwable e) { //异常通知 System.out.println("方法: " + methodName + " 异常 " + e); } finally { //后置通知 System.out.println("这是环绕通知之后的: " + methodName + " 结束"); } return result; } /** * 异常通知 */ @AfterThrowing(pointcut = "MyAdvice.pc()") public void afterException() { System.out.println("出现异常了: "); } /** * 后置通知(如果出现异常也会调用) */ @After("execution(* com.spring.service.*ServiceImpl.*(..))") public void after(JoinPoint joinPoint) { System.out.println("---"); System.out.println("拦截的方法名称: " + joinPoint.getSignature().getName()); System.out.println("这是后置通知(如果出现异常也会调用)"); } } ~~~ # 进行配置 ~~~ <aop:config>来声明要对aop进行配置 <aop:pointcut>它是用于声明切点(简单说就是对哪些方法进行拦截) <aop:advisor> 定义传统的aop的切面,传统的aop切面它只能包含一个切点与一个增强 <aop:aspect>定义aspectj框架的切面.,它可以包含多个切点与多个通知 ~~~ ~~~ <!-- 准备工作: 导入aop(约束)命名空间 --> <!-- 1.配置目标对象 --> <bean name="userService" class="service.UserServiceImpl"></bean> <!-- 2.配置通知对象 --> <bean name="myAdvice" class="aspect.MyAdvice"></bean> <!-- 3.配置将通知织入目标对象 --> <aop:config> <!-- 配置切入点 public void service.UserServiceImpl.save() void service.UserServiceImpl.save() * service.UserServiceImpl.save() * service.UserServiceImpl.*() * service.UserServiceImpl.*(..) * service.*ServiceImpl.*(..) * service..*ServiceImpl.*(..) //还会找子包 --> <aop:pointcut expression="execution(* service.*ServiceImpl.*(..))" id="pc"/> <aop:aspect ref="myAdvice"> <!-- 指定名为before方法为前置通知 --> <aop:before method="before" pointcut-ref="pc" /> <!-- 后置通知(如果出现异常不会调用) --> <aop:after-returning method="afterReturning" pointcut-ref="pc" returning="val" /> <!-- 环绕通知 --> <aop:around method="around" pointcut-ref="pc" /> <!-- 异常拦截通知 --> <aop:after-throwing method="afterException" pointcut-ref="pc" /> <!-- 后置 --> <aop:after method="after" pointcut-ref="pc" /> </aop:aspect> </aop:config> </beans> ~~~ ~~~ 关于execution语法常用: 1. execution(public * *()) 所有的public的方法 2. execution(* cn.study.aop.*(..)) 所有的aop包下的所有类的方法(不包含子包) 3. execution(* cn.study.aop..*(..)) 所有的aop包及其子包下的所有类的方法 4. execution(* cn.study.aop.IOrderService.*(..)) IOrderService接口中定义的所有方法 5. execution(* cn.study.aop.IOrderService+.*(..)) 匹配实现特定接口所有类的方法 6. execution(* save*(..)) 区配所有的以save开头的方法 ~~~ # 测试 ~~~ //帮我们创建容器 @RunWith(SpringJUnit4ClassRunner.class) // 指定创建容器时使用那个配置文件 @ContextConfiguration("classpath:applicationContext.xml") public class Demo { @Resource(name = "userService") private UserService us; @Test public void fun1() { us.save("hello"); } } ~~~ 结果 ~~~ 这是环绕通知之前的部分 --- 拦截的目标类: com.spring.service.UserServiceImpl 拦截的方法名称: save 输入的参数: [hello] 这是前置通知 保存用户--- 这是环绕通知之后的部分 --- 拦截的方法名称: save 这是后置通知(如果出现异常也会调用) --- 返回值: 1 这是后置通知(如果出现异常不会调用) ~~~