0

0

Spring Boot中创建非单例(原型)Bean:解决有状态对象实例隔离问题

心靈之曲

心靈之曲

发布时间:2025-10-20 09:13:00

|

702人浏览过

|

来源于php中文网

原创

Spring Boot中创建非单例(原型)Bean:解决有状态对象实例隔离问题

本文探讨了在spring boot应用中如何创建非单例(原型)作用域的bean。默认情况下,spring的@bean方法会生成单例实例,但这对于像resttemplatebuilder等有状态对象会导致副作用。通过使用@scope("prototype")注解,开发者可以确保每次注入请求都获得一个全新的bean实例,从而有效隔离对象状态,避免意外的全局影响。

Spring Boot Bean的默认行为与挑战

在Spring Boot应用中,当我们使用@Configuration类中的@Bean注解来定义一个Bean时,Spring IoC容器默认会将其注册为单例(Singleton)作用域。这意味着无论有多少个其他组件通过@Autowired请求这个Bean,它们都将共享同一个实例。这种设计对于无状态的服务组件(如Service、Repository等)非常高效,因为它减少了对象的创建和垃圾回收开销。

然而,对于某些有状态(Stateful)的对象,单例模式可能会引入意想不到的副作用。一个典型的例子是RestTemplateBuilder。RestTemplateBuilder是一个用于构建RestTemplate实例的工具类,它允许我们在构建RestTemplate时配置拦截器、消息转换器、超时设置等。由于RestTemplateBuilder内部维护着构建RestTemplate所需的状态信息,如果多个组件共享同一个RestTemplateBuilder实例,并且其中一个组件修改了它的配置(例如添加了一个新的拦截器),那么这些修改将会影响到所有其他使用该RestTemplateBuilder实例的组件,这往往不是我们期望的行为,可能导致难以调试的全局副作用。在这种情况下,我们需要为每个注入请求提供一个独立的、全新的Bean实例。

引入原型(Prototype)作用域

为了解决上述问题,Spring框架提供了多种Bean作用域,其中“原型”(Prototype)作用域正是为这种需求而设计的。当一个Bean被定义为原型作用域时,Spring IoC容器在每次收到该Bean的请求时,都会创建一个全新的实例并返回。这意味着每次通过@Autowired注入该Bean时,或者通过ApplicationContext.getBean()方法获取该Bean时,都会得到一个独立的、互不影响的对象。

要将一个默认的单例Bean修改为原型作用域,我们只需在@Bean注解旁边添加@Scope("prototype")注解即可。

实现原型Bean的代码示例

以下是如何在Spring Boot应用中定义一个原型作用域的Bean的示例:

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Scope;
import org.springframework.beans.factory.config.ConfigurableBeanFactory;

@Configuration
public class AppConfig {

    // 示例:定义一个原型作用域的Person Bean
    // 每次注入Person对象时,都会创建一个新的Person实例
    @Bean
    @Scope("prototype")
    public Person personPrototype() {
        return new Person("Prototype Person");
    }

    // 示例:使用常量定义原型作用域,提高可读性和类型安全
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    public MyService myServicePrototype() {
        return new MyService("Prototype Service");
    }
}

// 假设我们有一个简单的Person类
class Person {
    private String name;

    public Person(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

// 假设我们有一个简单的MyService类
class MyService {
    private String description;

    public MyService(String description) {
        this.description = description;
    }

    public String getDescription() {
        return description;
    }
}

在上述代码中,personPrototype()方法和myServicePrototype()方法都通过@Scope("prototype")注解明确指定了它们返回的Bean是原型作用域。当其他组件注入Person或MyService时,它们将获得各自独立的实例。

你也可以使用ConfigurableBeanFactory.SCOPE_PROTOTYPE常量来代替字符串字面量"prototype",这在一定程度上可以提高代码的可读性和类型安全性。

FreeTTS
FreeTTS

FreeTTS是一个免费开源的在线文本到语音生成解决方案,可以将文本转换成MP3,

下载

原型Bean的工作原理与适用场景

原型Bean的核心在于“按需创建”。当Spring容器收到一个对原型Bean的请求时,它不会从缓存中返回一个现有实例,而是执行以下步骤:

  1. 调用相应的@Bean方法(或实例化类)。
  2. 对新创建的实例执行依赖注入(如果Bean有依赖)。
  3. 返回这个全新的、完全独立的实例。

这种机制确保了每个消费者都能获得一个“新鲜”的对象,其状态不会受到其他消费者操作的影响。

适用场景:

  • 有状态的工具类或构建器: 如前文所述的RestTemplateBuilder,或者任何需要进行配置并可能在配置过程中修改内部状态的构建器模式对象。
  • 需要独立生命周期的对象: 如果一个对象在被使用后需要被“丢弃”或其状态不应影响其他实例,原型作用域是理想选择。
  • 线程不安全的组件: 对于那些内部状态在多线程环境下不安全的组件,如果不能通过其他同步机制解决,为每个线程或每次请求提供一个原型实例可以避免并发问题。

注意事项

虽然原型作用域解决了单例模式下有状态对象的隔离问题,但在使用时也需要注意以下几点:

  1. 生命周期管理: 与单例Bean不同,Spring容器在创建原型Bean后,不会对其进行后续的生命周期管理。这意味着,如果原型Bean中定义了@PreDestroy方法或实现了DisposableBean接口,Spring容器将不会调用它们来执行销毁逻辑。开发者需要自行管理原型Bean的销毁,或者确保它们不会持有需要显式释放的资源。
  2. 性能开销: 每次请求一个原型Bean都会导致一个新的对象实例被创建,这会带来一定的CPU和内存开销。如果原型Bean的创建成本很高,或者被频繁请求,这可能会对应用性能产生影响。因此,在决定使用原型作用域时,需要权衡其带来的隔离性优势与潜在的性能成本。
  3. 依赖注入的限制: 如果一个单例Bean依赖于一个原型Bean,那么该单例Bean在初始化时只会注入一个原型Bean的实例。之后,即使原型Bean被修改,该单例Bean也不会获得新的原型实例。如果单例Bean需要每次都获取一个新的原型实例,它需要通过ApplicationContext编程方式获取,或者使用方法注入(Method Injection)等高级特性。

总结

在Spring Boot应用中,理解并合理运用Bean的作用域是构建健壮、可维护应用的关键。当默认的单例作用域无法满足有状态对象隔离的需求时,@Scope("prototype")提供了一个强大而灵活的解决方案。通过将其应用于@Bean方法,我们可以确保每次注入都获得一个全新的、独立的Bean实例,从而有效避免因状态共享而导致的副作用。然而,开发者也应牢记原型Bean的生命周期管理特性和潜在的性能开销,以便做出最适合应用场景的设计选择。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

106

2025.08.06

spring boot框架优点
spring boot框架优点

spring boot框架的优点有简化配置、快速开发、内嵌服务器、微服务支持、自动化测试和生态系统支持。本专题为大家提供spring boot相关的文章、下载、课程内容,供大家免费下载体验。

135

2023.09.05

spring框架有哪些
spring框架有哪些

spring框架有Spring Core、Spring MVC、Spring Data、Spring Security、Spring AOP和Spring Boot。详细介绍:1、Spring Core,通过将对象的创建和依赖关系的管理交给容器来实现,从而降低了组件之间的耦合度;2、Spring MVC,提供基于模型-视图-控制器的架构,用于开发灵活和可扩展的Web应用程序等。

390

2023.10.12

Java Spring Boot开发
Java Spring Boot开发

本专题围绕 Java 主流开发框架 Spring Boot 展开,系统讲解依赖注入、配置管理、数据访问、RESTful API、微服务架构与安全认证等核心知识,并通过电商平台、博客系统与企业管理系统等项目实战,帮助学员掌握使用 Spring Boot 快速开发高效、稳定的企业级应用。

69

2025.08.19

Java Spring Boot 4更新教程_Java Spring Boot 4有哪些新特性
Java Spring Boot 4更新教程_Java Spring Boot 4有哪些新特性

Spring Boot 是一个基于 Spring 框架的 Java 开发框架,它通过 约定优于配置的原则,大幅简化了 Spring 应用的初始搭建、配置和开发过程,让开发者可以快速构建独立的、生产级别的 Spring 应用,无需繁琐的样板配置,通常集成嵌入式服务器(如 Tomcat),提供“开箱即用”的体验,是构建微服务和 Web 应用的流行工具。

34

2025.12.22

Java Spring Boot 微服务实战
Java Spring Boot 微服务实战

本专题深入讲解 Java Spring Boot 在微服务架构中的应用,内容涵盖服务注册与发现、REST API开发、配置中心、负载均衡、熔断与限流、日志与监控。通过实际项目案例(如电商订单系统),帮助开发者掌握 从单体应用迁移到高可用微服务系统的完整流程与实战能力。

114

2025.12.24

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1491

2023.10.24

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

278

2023.08.03

菜鸟裹裹入口以及教程汇总
菜鸟裹裹入口以及教程汇总

本专题整合了菜鸟裹裹入口地址及教程分享,阅读专题下面的文章了解更多详细内容。

0

2026.01.22

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Java 教程
Java 教程

共578课时 | 49.5万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号