SpringBoot当中的Singleton和Prototype详解

在Spring Boot中,Singleton和Prototype是两种Bean的作用域。这两种作用域决定了Spring容器如何创建和管理Bean的实例。

  1. Singleton(单例):

    • 当一个Bean被配置为Singleton作用域时,Spring容器在启动时只会创建该Bean的一个实例,并在后续的所有请求中共享这个单例实例。
    • 这是默认的作用域,如果你没有明确指定Bean的作用域,那么Spring会默认将其作为Singleton处理。
    • 对于无状态或者线程安全的Bean,通常选择Singleton作用域,因为这样可以提高性能并减少内存消耗。
  2. Prototype(原型):

    • 当一个Bean被配置为Prototype作用域时,每次通过Spring容器请求该Bean时,都会创建一个新的实例。
    • 这意味着对于每个请求或依赖注入,都会生成一个新的对象,每个对象之间是独立的,修改一个实例的状态不会影响到其他实例。
    • 对于有状态或者非线程安全的Bean,通常选择Prototype作用域,以确保每个请求都有自己的独立对象。

在Spring Boot中,你可以通过以下方式在Java配置或XML配置中指定Bean的作用域:

Java配置示例:

java 复制代码
@Bean
@Scope("singleton") // 默认值,可以省略
public MySingletonBean mySingletonBean() {
    return new MySingletonBean();
}

@Bean
@Scope("prototype")
public MyPrototypeBean myPrototypeBean() {
    return new MyPrototypeBean();
}

XML配置示例:

xml 复制代码
<bean id="mySingletonBean" class="com.example.MySingletonBean" scope="singleton" />

<bean id="myPrototypeBean" class="com.example.MyPrototypeBean" scope="prototype" />

总的来说,Spring Boot中的Singleton和Prototype作用域与Spring框架中的行为是一致的,它们主要影响了Bean的实例化策略和生命周期管理。

  1. 类上修饰@Scope和方法上修饰@Scope的区别

在Spring中,@Scope注解用于定义Bean的作用域。当这个注解应用在类或方法上时,它们的主要区别在于作用的范围和影响的对象创建方式:

类上修饰@Scope:

当@Scope注解应用于类级别时,它会影响该类的所有实例。

例如,如果你将一个类标记为@Scope("prototype"),那么每次请求或依赖注入该类的实例时,Spring容器都会创建一个新的对象。

方法上修饰@Scope(这种方法级别的作用域通常被称为"方法作用域"或"Scoped Proxy"):

当@Scope注解应用于方法级别时,它的效果通常与AOP(面向切面编程)代理一起使用。

在这种情况下,Spring会为这个方法创建一个代理对象,然后根据指定的作用域策略来管理这个代理对象的生命周期。

例如,如果你在一个配置类的方法上使用@Scope("request"),那么每当一个新的HTTP请求到达时,Spring会为这个方法创建一个新的代理对象,并在这个请求的范围内共享这个对象。

需要注意的是,方法级别的@Scope通常不直接应用于业务类,而是应用于配置类中的@Bean方法。这是因为业务类通常不直接由Spring容器管理,而是通过配置类中的@Bean方法来定义和初始化。

总结起来,类上的@Scope影响整个类的所有实例,而方法上的@Scope主要用于定义特定方法的行为,并且通常与AOP代理一起工作,以实现更细粒度的作用域控制。在实际使用中,方法级别的作用域相对较少见,大多数情况下,我们会在类级别或者@Bean方法级别上使用@Scope注解。

  1. Spring注入的类重复调用对应的方法是活创建多个实例还是永远是一个实例?

在Spring中,对象的创建和管理是由Spring容器负责的。对于注入的类,如果它们被配置为Singleton作用域(这是默认的作用域),那么无论你如何多次调用该类的方法,Spring容器只会创建一个实例,并在所有需要的地方共享这个单例实例。

如果你将类配置为Prototype作用域,那么每次请求都会创建一个新的实例。在这种情况下,即使你调用的是同一个方法,但如果是在不同的实例上进行的,那么该方法就会被多个类实例分别调用。

所以,是否创建多个类实例取决于你在Spring配置中设置的作用域。对于Singleton作用域,永远都是同一个类实例;而对于Prototype作用域,每次请求都会创建一个新的类实例。

  1. 如果修改类当中的变量的值。其他地方调用这个值会变吗

这取决于类的作用域和变量的访问修饰符。

如果类被配置为Singleton作用域,并且变量是实例变量(非static),那么当在一个地方修改该变量的值时,其他地方调用这个变量时会看到更改后的值,因为整个应用中只有一个共享的实例。

如果类被配置为Prototype作用域,每次请求都会创建一个新的实例。因此,如果在一个实例中修改变量的值,其他实例中的相同变量不会受到影响,因为它们是独立的对象。

对于静态变量(static),无论类的作用域如何,所有实例共享同一个静态变量。所以,如果在一个地方修改静态变量的值,所有地方调用这个静态变量时都会看到更改后的值。

对于局部变量(在方法内部定义的变量),它们只在该方法的上下文中存在,修改局部变量的值不会影响到其他方法或实例中的变量。

总的来说,变量的可见性和变化范围取决于它的作用域(静态或实例)、类的作用域(Singleton或Prototype)以及访问修饰符(public、private、protected等)。

相关推荐
云徒川1 小时前
【设计模式】原型模式
java·设计模式·原型模式
QTX187302 小时前
JavaScript 中的原型链与继承
开发语言·javascript·原型模式
暮乘白帝过重山2 小时前
Singleton和Prototype的作用域与饿汉式/懒汉式的初始化方式
spring·原型模式·prototype·饿汉式·singleton·懒汉式
腥臭腐朽的日子熠熠生辉3 小时前
解决maven失效问题(现象:maven中只有jdk的工具包,没有springboot的包)
java·spring boot·maven
绝顶少年3 小时前
Component 与 Bean 的深度解析:详细讲解Component和bean的异同以及与其搭配使用的其他注解及其使用场景
原型模式
绝顶少年4 小时前
Spring Boot 注解:深度解析与应用场景
java·spring boot·后端
西木风落5 小时前
springboot整合Thymeleaf web开发出现Whitelabel Error Page
spring boot·thymeleaf error·whitelabelerror
有来技术6 小时前
从0到1手撸企业级权限系统:基于 youlai-boot(开源) + Java17 + Spring Boot 3 完整实战
java·spring boot·后端
橘猫云计算机设计7 小时前
基于springboot微信小程序的旅游攻略系统(源码+lw+部署文档+讲解),源码可白嫖!
java·spring boot·后端·微信小程序·毕业设计·旅游
卡戎-caryon7 小时前
【Linux网络与网络编程】03.UDP Socket编程
linux·服务器·网络·笔记·单例模式·udp·网络通信