Spring-AOP不生效之内部方法调用

Spring-AOP

现象 :接口实现类中有两个,doMethod1()调用了doMethod2(),此时对于AOP,如果现在外部对象调用doMethod1()方法的时候,会发现只有doMethod1()方法执行被拦截,AOP生效,而doMethod1()内部调用doMethod2()时并没有被拦截;外部对象单独调用doMethod1()时会被拦截。

java 复制代码
@Service
public class JdkProxyDemoServiceImpl implements IJdkProxyService {

    @Override
    public void doMethod1() {
         doMethod2();
        System.out.println("JdkProxyServiceImpl.doMethod1()");
    }

    @Override
    public String doMethod2() {
        System.out.println("JdkProxyServiceImpl.doMethod2()");
        return "hello world";
    }
}

分析: 拦截器的实现原理就是动态代理,实现AOP机制。Spring的代理实现有两种:一是基于 JDK Dynamic Proxy 技术而实现的;二是基于 CGLIB 技术而实现的。如果目标对象实现了接口,在默认情况下Spring会采用JDK的动态代理实现AOP

java 复制代码
//JDK动态代理生成的JdkProxyDemoServiceImpl 的代理类
@Service
public class JdkProxyDemoServiceProxy implements IJdkProxyService {
    private  IJdkProxyService  iJdkProxyService;
    
    public void setIJdkProxyService(IJdkProxyService iJdkProxyService) {  
        this.iJdkProxyService= iJdkProxyService;  
    }  
 
  
    public void doMethod1() {
       //前置通知
        doBefore() 
       iJdkProxyService.doMethod1();
    }

  
    public String doMethod2() {
       //前置通知
        doBefore() 
       return iJdkProxyService.doMethod2();
    }
 
      private void doBefore() { 
        System.out.println(" 前置通知...");  
    }  
}

当使用时,从IOC容器中获取的Bean对象都是代理对象,而不是Bean对象本身,由于this关键字应用的并不是该Bean对象的对象,而是其本身,因此此时Spring AOP是不能拦截到这些被嵌套调用的方法的。

java 复制代码
//当获取Bean对象使用时

//spring容器创建代理对象
IJdkProxyService  iJdkProxyService = new IJdkProxyServiceImpl();

JdkProxyDemoServiceProxy  serviceProxy  = new JdkProxyDemoServiceProxy ()
serviceProxy.setIJdkProxyService(iJdkProxyService);
IJdkProxyService iJdkProxyService = (IJdkProxyService) serviceProxy;


iJdkProxyService.doMethod1();

解决:

1、修改类,把内部自调用改掉。

2、将this.doMethod2()替换为:((IJdkProxyService ) AopContext.currentProxy()).doMethod2();此时需要修改springaop配置:

// 指示是否创建基于子类(CGLIB)的代理,而不是创建基于标准Java接口的代理。 默认值是{@code false}。

@EnableAspectJAutoProxy(proxyTargetClass = true)

3、使用SpringUtil.getBean("iJdkProxyService ").doMethod2();

java 复制代码
@Component
public class SpringUtil implements ApplicationContextAware {
    private static ApplicationContext applicationContext;
    
    public SpringUtil springUtil() {
        return new SpringUtil();
    }
    
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        if(SpringUtil.applicationContext == null) {
            SpringUtil.applicationContext = applicationContext;
        }
    }

    //获取applicationContext
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    //通过name获取 Bean.
    public static Object getBean(String name){
        return getApplicationContext().getBean(name);
    }

    //通过class获取Bean.
    public static <T> T getBean(Class<T> clazz){
        return getApplicationContext().getBean(clazz);
    }

    //通过name,以及Clazz返回指定的Bean
    public static <T> T getBean(String name,Class<T> clazz){
        return getApplicationContext().getBean(name, clazz);
    }

}
相关推荐
zaim142 分钟前
计算机的错误计算(一百一十四)
java·c++·python·rust·go·c·多项式
hong_zc2 小时前
算法【Java】—— 二叉树的深搜
java·算法
进击的女IT3 小时前
SpringBoot上传图片实现本地存储以及实现直接上传阿里云OSS
java·spring boot·后端
Miqiuha3 小时前
lock_guard和unique_lock学习总结
java·数据库·学习
杨半仙儿还未成仙儿4 小时前
Spring框架:Spring Core、Spring AOP、Spring MVC、Spring Boot、Spring Cloud等组件的基本原理及使用
spring boot·spring·mvc
一 乐4 小时前
学籍管理平台|在线学籍管理平台系统|基于Springboot+VUE的在线学籍管理平台系统设计与实现(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端·学习
数云界4 小时前
如何在 DAX 中计算多个周期的移动平均线
java·服务器·前端
阑梦清川4 小时前
Java继承、final/protected说明、super/this辨析
java·开发语言
快乐就好ya6 小时前
Java多线程
java·开发语言
IT学长编程6 小时前
计算机毕业设计 二手图书交易系统的设计与实现 Java实战项目 附源码+文档+视频讲解
java·spring boot·毕业设计·课程设计·毕业论文·计算机毕业设计选题·二手图书交易系统