相关推荐recommended
Spring高手之路-Spring的AOP失效场景详解
作者:mmseoamin日期:2024-01-21

目录

前言

1.非Spring管理的对象

2.同一个Bean内部方法调用

3.静态方法

4.final方法

5.异步方法

总结


前言

Spring的AOP(面向切面编程)是一种强大的技术,用于在应用程序中实现横切关注点的模块化。虽然Spring的AOP在大多数情况下都是有效的,但在某些场景下可能会失效。下面来分析Spring AOP失效的常见场景

关于什么是Spring Aop

Spring高手之路-Spring AOP-CSDN博客Spring高手之路-Spring的AOP失效场景详解,icon-default.png?t=N7T8,第1张https://blog.csdn.net/qq_62262918/article/details/135197938?spm=1001.2014.3001.5502

首先,Spring的AOP其实是通过动态代理实现的,所以,想要让AOP生效,前提必须是动态代理生效,并且可以调用到代理对象的方法

1.非Spring管理的对象

Spring的AOP只能拦截由Spring容器管理的Bean对象。如果您使用了非受Spring管理的对象,则AOP将无法对其进行拦截。

2.同一个Bean内部方法调用

如果一个Bean内部的方法直接调用同一个Bean内部的另一个方法,AOP将无法拦截这个内部方法调用。因为AOP是基于代理的,只有通过代理对象才能触发AOP拦截。

@Component
public class MyBean {
  
  public void method1() {
    System.out.println("Inside method1");
    method2(); // 直接调用同一个 Bean 内部的另一个方法
  }
  
  public void method2() {
    System.out.println("Inside method2");
  }
}

假设我们有一个名为 MyBean 的类,其中包含了两个方法 method1() 和 method2()。在 method1() 中,直接调用了 method2() 方法。

现在,让我们创建一个切面来拦截 method1() 的执行,并打印一些日志信息:

@Aspect
@Component
public class MyAspect {
  
  @Before("execution(* com.example.MyBean.method1())")
  public void beforeMethod1() {
    System.out.println("Before method1 execution");
  }
}

上述切面使用 @Before 注解来定义了一个前置通知,在执行 MyBean 类的 method1() 方法之前被触发。

然后,我们在 Spring 应用程序中使用这两个组件:

@SpringBootApplication
public class MyApplication {
  
  public static void main(String[] args) {
    SpringApplication.run(MyApplication.class, args);
    
    MyBean myBean = ApplicationContext.getBean(MyBean.class);
    myBean.method1();
  }
}

当我们运行应用程序时,我们会发现 "Before method1 execution" 这条日志信息被打印出来,但是 "Inside method2" 这条日志信息却没有被打印出来。这是因为 AOP 无法拦截 method2() 的直接调用,而只能拦截通过代理对象触发的方法调用。

3.静态方法

Spring的AOP只能拦截非静态方法。如果您尝试拦截静态方法,AOP将无法生效。

4.final方法

AOP无法拦截final方法。final方法是不可重写的,因此AOP无法生成代理对象来拦截这些方法。

直接在对象内部调用方法:如果您直接在对象内部调用方法而不通过代理对象,AOP将无法拦截。因此,建议始终通过代理对象调用方法以确保AOP的生效。

5.异步方法

对于使用Spring的异步特性(如@Async注解)的方法,AOP拦截器可能无法正常工作。这是因为异步方法在运行时会创建新的线程或使用线程池,AOP拦截器无法跟踪到这些新线程中的方法调用。

总结

以下几种情况会导致AOP失效:

非Spring管理的对象

私有方法调用

静态方法调用

final方法调用

类内部自调用

内部类方法调用

异步方法