Spring 创建 Bean 的 8 种主要方式

Spring(尤其是 Spring Boot)提供了多种方式 来让容器创建和管理 Bean。@Component@Configuration + @Bean@EnableConfigurationProperties 都是常见方式。

下面我为你系统地梳理 Spring 创建 Bean 的所有主要方式,并说明它们的使用场景和区别。


✅ 一、Spring 创建 Bean 的 8 种主要方式

方式 注解/方法 适用场景
1 @Component 及其衍生注解 扫描普通类为 Bean
2 @Configuration + @Bean 手动配置第三方类或复杂对象
3 @EnableConfigurationProperties + @ConfigurationProperties 将配置文件映射为 Bean
4 @Import 导入配置类或直接注册 Bean
5 实现 FactoryBean<T> 创建复杂或条件性对象
6 使用 @Conditional 系列注解 条件化注册 Bean
7 编程式注册:BeanDefinitionRegistryPostProcessor 动态注册 Bean(高级)
8 XML 配置(传统方式) 老项目兼容

1. @Component 及其衍生注解(最常用)

java 复制代码
@Component
@Service
@Repository
@Controller
@RestController
@Configuration // @Configuration 也是 @Component
  • 原理 :配合 @ComponentScan,Spring 会扫描这些类并注册为 Bean。
  • 适用:你自己写的业务类(Service、DAO、Controller 等)。
java 复制代码
@Service
public class UserService {
    public void save() { ... }
}

✅ 自动注册为 userService Bean。


2. @Configuration + @Bean(手动注册)

用于注册不能加 @Component 的类(如第三方库的类)。

java 复制代码
@Configuration
public class RedisConfig {
    
    @Bean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {
        RedisTemplate<String, Object> template = new RedisTemplate<>();
        template.setConnectionFactory(factory);
        return template;
    }
}
  • 适用:数据源、RedisTemplate、RabbitMQ、RocketMQ、OkHttp、HttpClient 等第三方组件。

3. @EnableConfigurationProperties + @ConfigurationProperties

application.yml 中的配置自动绑定为 Bean。

yaml 复制代码
app:
  title: MyApp
  version: 1.0.0
java 复制代码
@Component
@ConfigurationProperties(prefix = "app")
public class AppProperties {
    private String title;
    private String version;
    // getter/setter
}
java 复制代码
@Configuration
@EnableConfigurationProperties(AppProperties.class)
public class AppConfig {
}

AppProperties 会被 Spring 创建并注入配置值,且可被 @Autowired
💡 在 Spring Boot 2.2+ 中,@EnableConfigurationProperties 通常可以省略,只要类加了 @ConfigurationProperties 并被扫描到即可。


4. @Import(导入配置类或直接注册)

三种用法:

(1) 导入 @Configuration
java 复制代码
@Import(DatabaseConfig.class)
@SpringBootApplication
public class Application { }
(2) 导入 @Component
java 复制代码
@Import(UserService.class)
(3) 导入 ImportSelectorImportBeanDefinitionRegistrar
java 复制代码
@Import(MyImportSelector.class)
public class Application { }

public class MyImportSelector implements ImportSelector {
    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        return new String[]{MyService.class.getName()};
    }
}

✅ 用于实现"条件导入"、"SPI 扩展"等高级功能。


5. 实现 FactoryBean<T>(创建复杂对象)

当你需要复杂的创建逻辑时使用。

java 复制代码
public class MyFactoryBean implements FactoryBean<MyService> {

    @Override
    public MyService getObject() throws Exception {
        MyService service = new MyService();
        service.init(); // 复杂初始化
        return service;
    }

    @Override
    public Class<?> getObjectType() {
        return MyService.class;
    }

    @Override
    public boolean isSingleton() {
        return true;
    }
}
java 复制代码
@Component
public class MyFactoryBean { } // Spring 会调用 getObject() 创建 MyService Bean

✅ 适合创建代理对象、动态对象、SPI 实现等。


6. @Conditional 系列注解(条件化注册)

只有满足条件时才创建 Bean。

java 复制代码
@Bean
@ConditionalOnMissingBean(DataSource.class)
public DataSource dataSource() {
    return new HikariDataSource();
}

常见条件注解:

注解 说明
@ConditionalOnClass 类路径存在某个类
@ConditionalOnMissingBean 容器中没有某个 Bean
@ConditionalOnProperty 配置文件中有某个属性
@ConditionalOnWebApplication 是 Web 应用
@ConditionalOnExpression SpEL 表达式为 true

✅ 这是 Spring Boot 自动配置的核心机制。


7. 编程式注册:BeanDefinitionRegistryPostProcessor

最强大、最灵活的方式,可以在容器启动时动态注册 Bean。

java 复制代码
@Component
public class DynamicBeanRegistrar implements BeanDefinitionRegistryPostProcessor {

    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
        BeanDefinitionBuilder builder = BeanDefinitionBuilder.rootBeanDefinition(UserService.class);
        AbstractBeanDefinition definition = builder.getBeanDefinition();
        registry.registerBeanDefinition("userService", definition);
    }

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        // do nothing
    }
}

✅ 适合插件化、动态加载、AOP 增强等场景。


8. XML 配置(传统方式)

xml 复制代码
<bean id="userService" class="com.example.UserService"/>
java 复制代码
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");

❌ 现在已不推荐,但在老项目中仍可见。


✅ 总结对比

方式 何时使用 推荐度
@Component 自己写的类 ⭐⭐⭐⭐⭐
@Configuration + @Bean 第三方类、复杂配置 ⭐⭐⭐⭐⭐
@ConfigurationProperties 配置文件映射 ⭐⭐⭐⭐
@Import 模块化导入 ⭐⭐⭐
FactoryBean 复杂对象创建 ⭐⭐⭐
@Conditional 条件注册 ⭐⭐⭐⭐(自动配置核心)
BeanDefinitionRegistryPostProcessor 动态注册 ⭐⭐(高级)
XML 老项目

✅ 最佳实践建议

  1. 优先使用 @Component@Bean,简单直接;
  2. 配置参数用 @ConfigurationProperties,类型安全;
  3. 避免重复注册 ,善用 @ConditionalOnMissingBean
  4. 复杂逻辑用 @ImportFactoryBean
  5. 动态场景用 BeanDefinitionRegistryPostProcessor(谨慎使用)。

相关推荐
Chan1617 小时前
流量安全优化:基于 Nacos 和 BloomFilter 实现动态IP黑名单过滤
java·spring boot·后端·spring·nacos·idea·bloomfilter
非凡ghost18 小时前
PixPin截图工具(支持截长图截动图) 中文绿色版
前端·javascript·后端
武子康18 小时前
大数据-133 ClickHouse 概念与基础|为什么快?列式 + 向量化 + MergeTree 对比
大数据·后端·nosql
脚踏实地的大梦想家18 小时前
【Go】P11 掌握 Go 语言函数(二):进阶玩转高阶函数、闭包与 Defer/Panic/Recover
开发语言·后端·golang
小小爱大王18 小时前
AI 编码效率提升 10 倍的秘密:Prompt 工程 + 工具链集成实战
java·javascript·人工智能
用户685453759776919 小时前
🔥 服务熔断降级:微服务的"保险丝"大作战!
后端
Tech有道19 小时前
拼多多「面试官问我:LRU 和 LFU 你选谁?」我:看场景啊哥!😂
后端
用户685453759776919 小时前
🎬 开场:RPC框架的前世今生
后端
王中阳Go背后的男人19 小时前
Docker磁盘满了?这样清理高效又安全
后端·docker