1.3 IoC容器核心机制

1.3 IoC容器核心机制

1.3.1 Bean生命周期全流程解析(深度源码级)

完整生命周期流程图

实例化Bean
属性填充
BeanNameAware.setBeanName
BeanFactoryAware.setBeanFactory
ApplicationContextAware.setApplicationContext
BeanPostProcessor.postProcessBeforeInitialization
@PostConstruct
InitializingBean.afterPropertiesSet
自定义init-method
BeanPostProcessor.postProcessAfterInitialization
Bean就绪
容器关闭
@PreDestroy
DisposableBean.destroy
自定义destroy-method

关键阶段代码演示

public class LifecycleBean implements 
    BeanNameAware, BeanFactoryAware, 
    ApplicationContextAware, InitializingBean, DisposableBean {

    private String beanName;
    
    // 1. Aware接口注入
    @Override
    public void setBeanName(String name) {
        this.beanName = name;
        System.out.println("BeanNameAware: "+name);
    }

    @Override
    public void setBeanFactory(BeanFactory beanFactory) {
        System.out.println("BeanFactoryAware注入完成");
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) {
        System.out.println("ApplicationContextAware注入完成");
    }

    // 2. 初始化回调
    @PostConstruct
    public void customInit() {
        System.out.println("@PostConstruct方法执行");
    }

    @Override
    public void afterPropertiesSet() {
        System.out.println("InitializingBean.afterPropertiesSet执行");
    }

    public void xmlInit() {
        System.out.println("XML定义的init-method执行");
    }

    // 3. 销毁回调
    @PreDestroy
    public void customDestroy() {
        System.out.println("@PreDestroy方法执行");
    }

    @Override
    public void destroy() {
        System.out.println("DisposableBean.destroy执行");
    }

    public void xmlDestroy() {
        System.out.println("XML定义的destroy-method执行");
    }
}

// 配置类
@Configuration
public class LifecycleConfig {
    @Bean(initMethod = "xmlInit", destroyMethod = "xmlDestroy")
    public LifecycleBean lifecycleBean() {
        return new LifecycleBean();
    }
    
    @Bean
    public static BeanPostProcessor customProcessor() {
        return new BeanPostProcessor() {
            @Override
            public Object postProcessBeforeInitialization(Object bean, String beanName) {
                System.out.println("BeanPostProcessor前置处理: "+beanName);
                return bean;
            }
            
            @Override
            public Object postProcessAfterInitialization(Object bean, String beanName) {
                System.out.println("BeanPostProcessor后置处理: "+beanName);
                return bean;
            }
        };
    }
}

控制台输出顺序

BeanNameAware: lifecycleBean
BeanFactoryAware注入完成
ApplicationContextAware注入完成
BeanPostProcessor前置处理: lifecycleBean
@PostConstruct方法执行
InitializingBean.afterPropertiesSet执行
XML定义的init-method执行
BeanPostProcessor后置处理: lifecycleBean
... [容器运行中]
@PreDestroy方法执行
DisposableBean.destroy执行
XML定义的destroy-method执行

1.3.2 依赖注入的六种实现方式(企业级最佳实践)

注入方式对照表

注入方式实现形式适用场景注意事项
构造器注入@Autowired构造函数强制依赖、不可变对象解决循环依赖的首选方式
Setter注入@Autowired set方法可选依赖、需要重新配置避免在并发场景下修改
字段注入@Autowired字段快速开发、简单场景不利于单元测试
方法注入@Bean工厂方法复杂对象创建逻辑需配合@Configuration使用
接口回调注入ApplicationContextAware获取容器基础设施增加代码耦合度
查找方法注入@Lookup注解原型Bean注入单例Bean需要CGLIB代理

构造器注入最佳实践

@Service
public class OrderService {
    private final PaymentService paymentService;
    private final InventoryService inventoryService;

    // 显式构造器注入
    @Autowired
    public OrderService(PaymentService paymentService, 
                       InventoryService inventoryService) {
        this.paymentService = paymentService;
        this.inventoryService = inventoryService;
    }
}

// 解决循环依赖方案
@Configuration
public class CircularDependencyConfig {
    @Bean
    @Lazy
    public ServiceA serviceA(ServiceB serviceB) {
        return new ServiceA(serviceB);
    }

    @Bean
    @Lazy
    public ServiceB serviceB(ServiceA serviceA) {
        return new ServiceB(serviceA);
    }
}

查找方法注入实战

@Component
@Scope("prototype")
public class PrototypeBean {
    private static int count = 0;
    private final int id;

    public PrototypeBean() {
        this.id = ++count;
    }

    public void printId() {
        System.out.println("PrototypeBean ID: " + id);
    }
}

@Component
public class SingletonBean {
    // 每次调用获得新的原型Bean
    @Lookup
    public PrototypeBean getPrototypeBean() {
        return null; // 实际由CGLIB实现
    }

    public void execute() {
        PrototypeBean bean = getPrototypeBean();
        bean.printId();
    }
}

1.3.3 自动装配的智能决策机制

自动装配冲突解决策略

发现多个候选Bean
是否有@Primary标记?
选择@Primary的Bean
是否有@Qualifier指定?
按限定符选择
是否开启宽松模式?
按名称匹配
抛出NoUniqueBeanDefinitionException

企业级配置案例

// 主数据源配置
@Configuration
public class PrimaryDataSourceConfig {
    @Bean
    @Primary
    public DataSource mainDataSource() {
        return DataSourceBuilder.create()
            .url("jdbc:mysql://master:3306/core")
            .username("admin")
            .password("secure123")
            .build();
    }

    @Bean
    public DataSource reportDataSource() {
        return DataSourceBuilder.create()
            .url("jdbc:mysql://replica:3306/report")
            .username("readonly")
            .password("readonly123")
            .build();
    }
}

// 业务服务层
@Service
public class ReportService {
    private final DataSource dataSource;

    // 明确指定非主数据源
    public ReportService(@Qualifier("reportDataSource") DataSource dataSource) {
        this.dataSource = dataSource;
    }
}

自动装配异常处理方案

// 1. 使用Optional避免依赖不存在
@Autowired
public void setOptionalDependency(Optional<SomeService> service) {
    service.ifPresent(s -> this.service = s);
}

// 2. 自定义缺失Bean处理
@Bean
@ConditionalOnMissingBean
public DefaultCacheManager defaultCache() {
    return new DefaultCacheManager();
}

// 3. 集合类型安全注入
@Autowired
public void setStrategies(List<BusinessStrategy> strategies) {
    this.strategyMap = strategies.stream()
        .collect(Collectors.toMap(
            s -> s.getClass().getSimpleName(), 
            Function.identity()
        ));
}

1.3.4 条件化Bean配置实战

多环境数据源切换方案

// 条件判断类
public class EnvCondition implements Condition {
    @Override
    public boolean matches(ConditionContext context, 
                          AnnotatedTypeMetadata metadata) {
        String env = context.getEnvironment()
                          .getProperty("app.env", "dev");
        return "prod".equals(env);
    }
}

// 生产环境配置
@Configuration
@Conditional(EnvCondition.class)
public class ProdDataSourceConfig {
    @Bean
    public DataSource prodDataSource() {
        HikariConfig config = new HikariConfig();
        config.setJdbcUrl("jdbc:mysql://prod-db:3306/core");
        config.setUsername("prod_user");
        config.setPassword(System.getenv("DB_PROD_PASS"));
        config.setConnectionTimeout(3000);
        return new HikariDataSource(config);
    }
}

// 开发环境配置
@Profile("dev")
@Configuration
public class DevDataSourceConfig {
    @Bean
    public DataSource devDataSource() {
        return new EmbeddedDatabaseBuilder()
            .setType(EmbeddedDatabaseType.H2)
            .addScript("classpath:schema.sql")
            .build();
    }
}

条件注解组合使用

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Conditional(OnCloudPlatformCondition.class)
@Conditional(OnKubernetesEnvCondition.class)
public @interface ConditionalOnCloudKubernetes {}

// 云原生环境专属配置
@Configuration
@ConditionalOnCloudKubernetes
public class CloudNativeConfig {
    @Bean
    public ServiceDiscovery serviceDiscovery() {
        return new KubernetesServiceDiscovery();
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/967334.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

傅里叶公式推导(一)

文章目录 三角函数系正交证明图观法数学证明法计算当 n不等于m当 n等于m&#xff08;重点&#xff09; 其它同理 首先要了解的一点基础知识&#xff1a; 三角函数系 { sin ⁡ 0 , cos ⁡ 0 , sin ⁡ x , cos ⁡ x , sin ⁡ 2 x , cos ⁡ 2 x , … , sin ⁡ n x , cos ⁡ n x ,…

SSA-TCN麻雀算法优化时间卷积神经网络时间序列预测未来Matlab实现

SSA-TCN麻雀算法优化时间卷积神经网络时间序列预测未来Matlab实现 目录 SSA-TCN麻雀算法优化时间卷积神经网络时间序列预测未来Matlab实现预测效果基本介绍程序设计参考资料 预测效果 基本介绍 1.Matlab实现SSA-TCN麻雀算法优化时间卷积神经网络时间序列预测未来&#xff08;优…

DeepSeek 助力 Vue 开发:打造丝滑的步骤条

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

利用二分法进行 SQL 盲注

什么是sql注入&#xff1f; SQL 注入&#xff08;SQL Injection&#xff09;是一种常见的 Web 安全漏洞&#xff0c;攻击者可以通过构造恶意 SQL 语句来访问数据库中的敏感信息。在某些情况下&#xff0c;服务器不会直接返回查询结果&#xff0c;而是通过布尔值&#xff08;Tr…

USB子系统学习(四)用户态下使用libusb读取鼠标数据

文章目录 1、声明2、HID协议2.1、描述符2.2、鼠标数据格式 3、应用程序4、编译应用程序5、测试6、其它 1、声明 本文是在学习韦东山《驱动大全》USB子系统时&#xff0c;为梳理知识点和自己回看而记录&#xff0c;全部内容高度复制粘贴。 韦老师的《驱动大全》&#xff1a;商…

数据安全:守护数据的坚固防线

在数字化时代&#xff0c;数据已成为企业和组织的核心资产。然而&#xff0c;数据的安全性问题也日益凸显&#xff0c;数据泄露、数据滥用等事件频发&#xff0c;给企业和个人带来了巨大的损失。今天&#xff0c;让我们深入《DAMA数据管理知识体系指南&#xff08;第二版&#…

PyQt学习记录

0. 安装配置 0.1 安装相关库 首先打开你的PyCharm程序&#xff0c;然后新建一个目录用于学习&#xff0c;其次在terminal中输入 pip install pyqt5如果你不具有科学上网能力&#xff0c;请改为国内源 pip install pyqt5 -i https://pypi.douban.com/simple然后安装pyqt相关…

对“云原生”的初印象

一、背景 最近因为在工作中以及一些技术博客中听的比较火的一个关键词 "云原生"&#xff0c;于是产生了好奇&#xff0c;云原生到底是什么东西&#xff1f;自己对云原生也是一个纯小白&#xff0c;于是带着这个问题去好好了解一下&#xff0c;什么是"云原生&qu…

SystemVerilog基础:disable fork语句

相关阅读 SystemVerilog基础https://blog.csdn.net/weixin_45791458/category_12517449.html?spm1001.2014.3001.5482 一、进程的概念 在学习disable fork语句之前&#xff0c;首先的了解SystemVerilog中的进程概念&#xff1a;进程是一系列可以独立执行的一个或多个表达式。…

富芮坤FR8003硬件:VDDIO供电有工作不正常的情况从VBAT供电正常

从事嵌入式单片机的工作算是符合我个人兴趣爱好的,当面对一个新的芯片我即想把芯片尽快搞懂完成项目赚钱,也想着能够把自己遇到的坑和注意事项记录下来,即方便自己后面查阅也可以分享给大家,这是一种冲动,但是这个或许并不是原厂希望的,尽管这样有可能会牺牲一些时间也有哪天原…

IBM服务器刀箱Blade安装Hyper-V Server 2019 操作系统

案例:刀箱某一blade,例如 blade 5 安装 Hyper-V Server 2019 操作系统(安装进硬盘) 刀箱USB插入安装系统U盘,登录192.168... IBM BlandeCenter Restart Blande 5,如果Restart 没反应,那就 Power Off Blade 然后再 Power On 重启后进入BIOS界面设置usb存储为开机启动项 …

【大模型】本地部署DeepSeek-R1:8b大模型及搭建Open-WebUI交互页面

本地部署DeepSeek-R1:8b大模型 一、摘要及版本选择说明1.1 摘要1.2 版本选择 二、下载并安装Ollama三、运行DeepSeek-R1:8b大模型四、安装Open WebUI增强交互体验五、关闭Ollama开机自动启动六、DeepSeek大模型启停步骤 一、摘要及版本选择说明 1.1 摘要 作为一名对 AI 和生成…

6、使用one-api管理统一管理大模型,并开始使用本地大模型

文章目录 本节内容介绍集中接入&#xff1a;将大模型统一管理起来当使用了大模型代理大模型代理示例 开源模型&#xff1a;如何使用Hugging Face上的模型modelscope使用 pipeline 调用模型用底层实现调用模型流式输出 如何在项目中使用开源模型使用 LangChain使用集中接入开始使…

绕组电感 - Ansys Maxwell 磁通链与电流

在本博客中&#xff0c;我将演示如何使用 Ansys Maxwell 中磁瞬态求解器的磁通链和电流结果来计算绕组电感。Ansys Maxwell 磁瞬态求解器在场计算中考虑了涡流效应&#xff0c;我将展示一种使用磁通链和电流结果来计算绕组电感的简单方法。 实际上&#xff0c;电感是非线性的…

【图片转换PDF】多个文件夹里图片逐个批量转换成多个pdf软件,子文件夹单独合并转换,子文件夹单独批量转换,基于Py的解决方案

建筑设计公司在项目执行过程中&#xff0c;会产生大量的设计图纸、效果图、实景照片等图片资料。这些资料按照项目名称、阶段、专业等维度存放在多个文件夹和子文件夹中。 操作需求&#xff1a;为了方便内部管理和向客户交付完整的设计方案&#xff0c;公司需要将每个项目文件…

Formality:探针(Probe Point)的设置与使用

相关阅读 Formalityhttps://blog.csdn.net/weixin_45791458/category_12841971.html?spm1001.2014.3001.5482 一般情况下&#xff0c;verify命令会对参考设计和实现设计所有匹配的比较点各自进行验证&#xff0c;但有些时候为了调试&#xff0c;可能需要验证参考设计和实现设…

Cherry Studio之DeepSeek联网/本地,建属于自己的AI助理!

上一篇文章&#xff0c;讲了DeepSeek-R1部署到本地的方法。这一篇文章&#xff0c;我们让DeepSeek再一次升级&#xff0c;通过图形化界面来交互&#xff0c;从而变成我们的AI助理&#xff0c;让DeepSeek R1发挥最大实力&#xff01; 首选需要借助硅基流动的API接口&#xff0c…

HarmonyOS Next 方舟字节码文件格式介绍

在开发中&#xff0c;可读的编程语言要编译成二进制的字节码格式才能被机器识别。在HarmonyOS Next开发中&#xff0c;arkts会编译成方舟字节码。方舟字节码长什么样呢&#xff1f;我们以一个demo编译出的abc文件&#xff1a; 二进制就是长这样&#xff0c;怎么去理解呢&…

AMD 8845HS 780M核显部署本地deepseek大模型的性能

测试了一下笔记本电脑AMD 8845HS的780M核显是否能本地部署deepseek大模型。 测试软件环境&#xff1a;LM Studio 0.3.9 、Windows 11 24H2 硬件&#xff1a;荣耀X16笔记本 CPU&#xff1a;AMD 8845HS 显卡&#xff1a;780M核显&#xff0c;显存为共享内存自动分配模式&…

实验7 路由器之间IPsec VPN配置

实验7 路由器之间IPsec VPN配置 1.实验目的 通过在两台路由器之间配置IPsec VPN连接&#xff0c;掌握IPsec VPN配置方法&#xff0c;加深对IPsec协议的理解。 2.实验内容 &#xff08;1&#xff09;按照实验拓扑搭建实验环境。 &#xff08;2&#xff09;在路由器R1和R4配置IP…