当前位置: 首页 > news >正文

@Configuration原理与实战

文章目录

  • 前言
  • 一、@Component与@Configuration区别
  • 二、@Configuration的解析
  • 三、生成CGLIB动态代理
  • 四、目标方法的调用
    • 4.1、BeanMethodInterceptor
  • 总结


前言

  @Configuration是Spring提供的注解,其作用是将该类标识为一个Java 配置类,用来替代传统的 XML 配置文件:

@Configuration
public class AppConfig {@Beanpublic MyService myService() {return new MyService();}
}

  上述代码等价于传统 XML 中:

<bean id="myService" class="com.example.MyService"/>

  与标注了@Component注解的配置类相比,区别在于Spring 在解析@Configuration 类时,会使用 CGLIB 生成子类代理,重写其中所有@Bean方法。


一、@Component与@Configuration区别

  @Configuration案例工程:

public class MainApp {public static void main(String[] args) {// 创建应用上下文AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);AppConfig appConfig = context.getBean(AppConfig.class);System.out.println("------ 执行 AppConfig 中的 testBeanMethodCall ------");appConfig.testBeanMethodCall(); // 使用的是代理对象,返回同一个 Beancontext.close();}
}class HelloService {}@Configuration
class AppConfig {@Beanpublic HelloService helloService() {return new HelloService();}public void testBeanMethodCall() {HelloService s1 = helloService();HelloService s2 = helloService();System.out.println("AppConfig -> helloService() s1 == s2 ? " + (s1 == s2));}
}

  运行结果:

------ 执行 AppConfig 中的 testBeanMethodCall ------
AppConfig -> helloService() s1 == s2 ? true

  @Component案例工程:

public class MainApp {public static void main(String[] args) {// 创建应用上下文AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);AppConfig appConfig = context.getBean(AppConfig.class);System.out.println("------ 执行 AppConfig 中的 testBeanMethodCall ------");appConfig.testBeanMethodCall(); // 使用的是代理对象,返回同一个 Beancontext.close();}
}class HelloService {}@Component
class AppConfig {@Beanpublic HelloService helloService() {return new HelloService();}public void testBeanMethodCall() {HelloService s1 = helloService();HelloService s2 = helloService();System.out.println("AppConfig -> helloService() s1 == s2 ? " + (s1 == s2));}
}

  运行结果:

------ 执行 AppConfig 中的 testBeanMethodCall ------
AppConfig -> helloService() s1 == s2 ? false


  通过最终的运行结果可以看出,被@Configuration修饰的配置类中,获取多次被@Bean注解标注的类,**获取到的都是相同的实例。**而被@Component修饰的配置类中,获取多次被@Bean注解标注的类,每次获取到的都是新的实例。

二、@Configuration的解析

  @Configuration的解析,同样在ConfigurationClassPostProcessor后置处理器中:
在这里插入图片描述
  如果当前配置类上,有@Configuration注解,那么会根据proxyBeanMethods的值,走不同的分支,proxyBeanMethods@Configuration注解的属性,默认为true:
在这里插入图片描述
  在配置类上加入了@Configuration注解,并且没有显式地标注proxyBeanMethods为false,则会将配置类的bean定义中的attributes的value设置为full,表示当前的配置类是一个完整的配置类:
在这里插入图片描述
在这里插入图片描述
  上述的过程是@Configuration的解析阶段

三、生成CGLIB动态代理

  ConfigurationClassPostProcessor后置处理器,实现了BeanDefinitionRegistryPostProcessor,而BeanDefinitionRegistryPostProcessor又继承了普通的bean工厂后处理器BeanFactoryPostProcessor
在这里插入图片描述
  在invokeBeanFactoryPostProcessorsinvokeBeanFactoryPostProcessors方法中,会调用ConfigurationClassPostProcessor重写的父类BeanFactoryPostProcessorpostProcessBeanFactory方法:
在这里插入图片描述

  收集所有bean定义的attributes的value为full的bean成map。key是bean的名称,value是对应的bean定义,在上面的案例工程中对应的就是AppConfig类。

att
  然后进行遍历,生成CGLIB动态代理
在这里插入图片描述
在这里插入图片描述
  这里设置的拦截器链,是定义在ConfigurationClassEnhancer中的一个属性:
在这里插入图片描述

private static final Callback[] CALLBACKS = new Callback[] {new BeanMethodInterceptor(),            // 拦截 @Bean 方法调用,核心代理逻辑new BeanFactoryAwareMethodInterceptor(),// 拦截实现 BeanFactoryAware 的方法NoOp.INSTANCE                          // 无操作回调,默认行为
};

  最后将AppConfig的类型设置为代理类型。

在这里插入图片描述

四、目标方法的调用

  在调用案例工程的testBeanMethodCall时,会走过BeanMethodInterceptor拦截器的intercept方法。
在这里插入图片描述

4.1、BeanMethodInterceptor

  如果容器是第一次调用@Bean的目标方法,就真正执行 @Bean 方法体:
在这里插入图片描述
  否则,直接从容器中获取已有的 bean:
在这里插入图片描述
在这里插入图片描述
关键代码,保证每次从容器中拿到的都是同一个单例实例

总结

特性@Configuration@Component
是否会被 Spring 扫描注册为 Bean
是否支持 @Bean 方法注册 Bean支持(推荐)支持(不推荐)
@Bean 方法是否被代理增强会(返回容器单例对象)不会(每次 new 新对象)
是否能作为配置中心使用推荐用于声明多个 @Bean不推荐作为配置类使用

相关文章:

  • ​计算机网络原理超详解说​
  • Web后端基础:Maven基础
  • Web 毕设篇-适合小白、初级入门练手的 Spring Boot Web 毕业设计项目:智驿AI系统(前后端源码 + 数据库 sql 脚本)
  • P4 QT项目----串口助手(4.2)
  • 2025 高考:AI 都在哪些地方发挥了作用
  • crackme007
  • 电脑一段时间没用就变成登陆的界面
  • CppCon 2015 学习:Implementing class properties effectively
  • RocketMQ延迟消息机制
  • 【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表
  • 第5章 类的基本概念 笔记
  • 不变性(Immutability)模式
  • b2b企业网络营销如何用deepseek、豆包等AI平台获客 上海添力
  • switch选择语句
  • 打造多模态交互新范式|彩讯股份中标2025年中国移动和留言平台AI智能体研发项目
  • Linux内核 -- INIT_WORK 使用与注意事项
  • Win系统下的Linux系统——WSL 使用手册
  • 如何根据excel表生成sql的insert脚本
  • [ICLR 2022]How Much Can CLIP Benefit Vision-and-Language Tasks?
  • PyArk飘云阁出品的ARK工具
  • 福州网站开发招聘/企业网站优化
  • 珠海企业建站程序/北京seo教师
  • 企业做网站得多少钱/网站流量查询
  • 哪个网站教做公众号/seo一键优化
  • 电子商务网站设计分析怎么做/重庆seo论
  • 手机网站怎么做单页面/外链生成工具