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

Spring-AOP

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

@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

//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是不能拦截到这些被嵌套调用的方法的。

//当获取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();

@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);
    }

}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring AOP未生效的原因可能有以下几种情况。 首先,如果要增强的类或方法没有被Spring管理,那么AOP就无法生效。这种情况下,需要将目标类或方法注入到Spring容器中才能实现AOP的增强效果。 其次,如果要被增强的类或方法同时被SpringSpring MVC扫描,可能会导致AOP失效。这是因为在Spring MVC容器中重新实例化的对象并没有被AOP代理,而是由Spring MVC容器直接管理。为了解决这个问题,我们需要确保AOP配置中的切点表达式声明与要切入的类和方法在同一个容器中。 另外,如果要增强的方法被同类的方法调用,也可能导致AOP不生效。这是因为AOP是在目标方法调用时才会生效,而同类的方法调用不会触发AOP。要解决这个问题,可以将目标方法抽取为单独的方法,然后在原来的方法和其他使用该方法的地方进行调用。 还有一种情况是,如果使用了注解方式进行AOP,但没有开启注解支持,也会导致AOP不生效。要解决这个问题,需要在Spring配置文件中添加注解支持的配置。 综上所述,要解决Spring AOP不生效的问题,需要注意以上几种情况,并根据具体情况进行相应的修改和调整。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [Spring Aop失效的情況及解决办法](https://blog.csdn.net/weixin_43790613/article/details/113091496)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值