spring aop 内部引用失效分析

背景

用了spring retry 发现在有些场景下失效,也让我想起了之前看spring 事务时会有一些场景下失效的这个问题

代码

bash 复制代码
接口

public interface UserService {


    public void start() ;


    public void stop();
}

实现类
@Service
public class UserServiceImpl implements UserService {
    @Override
    @Retryable()
    public void start() {

        if (1 == 1) {
            throw new RuntimeException();
        }

    }

    @Override
    public void stop() {

        start();
        

    }
}

测试

bash 复制代码
        ConfigurableApplicationContext run = SpringApplication.run(K8sDemoApplication.class, args);

        UserService user = run.getBean(UserService.class);
        
        //生效
        user.start();

        
        //失效
        // user.stop();

测试发现如果直接调用start 方法是会生效的

通过stop()方法里面调用start 方法是会失效的

分析

首先我们知道如果要在一个方法前后加一些todo ,有两种方法,一种是硬code

还有一种是通过动态代理的方法,这里通过注解的方法,实际底层是通过cglib这种代理实现的,默认应该是jdk动态代理的,但是我用arthas分析实际是用cglib这种形式,这个不重要。

为啥通过代理的,两种不同的调用方式会有区别呢

直接调用为啥是可以的,因为实际上的调用对象是生成的动态代理对象,在方法上进行代理的。

bash 复制代码

我们重点了解下为啥间接调用失败。

看下编译后的代码

bash 复制代码
@Service
public class UserServiceImpl implements UserService {
    public UserServiceImpl() {
    }

    public void start() {
        throw new RuntimeException();
    }

    public void stop() {
    //实际上调用start,我们可以发现是通过this 对是本身这个对象直接调用的
        this.start();
    }
}

总结

加深了对之前glibc动态代理的理解

相关推荐
Nobody_Cares24 分钟前
JWT令牌
java
沐浴露z25 分钟前
Kafka入门:基础架构讲解,安装与使用
java·分布式·kafka
神秘的土鸡30 分钟前
从数据仓库到数据中台再到数据飞轮:我的数据技术成长之路
java·服务器·aigc·数据库架构·1024程序员节
vir0243 分钟前
P1928 外星密码(dfs)
java·数据结构·算法·深度优先·1024程序员节
摇滚侠1 小时前
全面掌握PostgreSQL关系型数据库,备份和恢复,笔记46和笔记47
java·数据库·笔记·postgresql·1024程序员节
掘金码甲哥1 小时前
两张大图一次性讲清楚k8s调度器工作原理
后端
间彧2 小时前
Stream flatMap详解与应用实战
后端
间彧2 小时前
Java Stream流两大实战陷阱:并行流Parallel误用、List转Map时重复键异常
后端
eguid_12 小时前
【开源项目分享】JNSM1.2.0,支持批量管理的jar包安装成Windows服务可视化工具,基于Java实现的支持批量管理已经安装服务的可视化工具
java·开源·jar·1024程序员节·windows服务·jar包安装成服务·exe安装成服务
杯莫停丶3 小时前
设计模式之:享元模式
java·设计模式·享元模式