DevNotes-Spring的Controller是单例还是多例?-《Java笔记》

admin 2025-10-19 04:31:38 编程 来源:ZONE.CI 全球网 0 阅读模式

Java Spring Controller测试代码示例:

  1. @Controller
  2. public class ScopeTestController {
  3. private int num = 0;
  4. @RequestMapping("/testScope")
  5. public void testScope() {
  6. System.out.println(++num);
  7. }
  8. @RequestMapping("/testScope2")
  9. public void testScope2() {
  10. System.out.println(++num);
  11. }
  12. }

首先访问 http://localhost:8080/testScope,得到的答案是 1;然后再访问 http://localhost:8080/testScope2,得到的答案是 2。

得到的不同的值,这是线程不安全的。

在 ScopeTestController 类上加了一个注解 @Scope("prototype")

  1. import org.springframework.context.annotation.Scope;
  2. import org.springframework.stereotype.Controller;
  3. import org.springframework.web.bind.annotation.RequestMapping;
  4. /**
  5. * @date 2019/07/29 22:56
  6. */
  7. @Controller
  8. @Scope("prototype")
  9. public class ScopeTestController {
  10. private int num = 0;
  11. @RequestMapping("/testScope")
  12. public void testScope() {
  13. System.out.println(++num);
  14. }
  15. @RequestMapping("/testScope2")
  16. public void testScope2() {
  17. System.out.println(++num);
  18. }
  19. }

再次进行重复的验证 http://localhost:8080/testScope,得到的答案是 1。再访问 http://localhost:8080/testScope2,得到的答案还是 1。

解决方案

  • 不要在controller中定义成员变量。
  • 万一必须要定义一个非静态成员变量时候,则通过注解@Scope("prototype"),将其设置为多例模式。
  • 在Controller中使用ThreadLocal变量

    补充说明

    Spring Bean作用域有以下5个:**singleton**: 单例模式,当Spring创建applicationContext容器的时候,Spring会初始化所有的该作用域实例,加上lazy-init就可以避免预处理;**prototype** 原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后Spring将不再对其管理;(下面是在web项目下才用到的)**request** 搞web的大家都应该明白request的域了吧,就是每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,Spring依然在监听;**session**: 每次会话,同上;global session: 全局的web域,类似于servlet中的application。为什么加上 @Scope("prototype") 注解之后同样的验证方式就得出了不一样的结论呢?其实真正的答案就藏在 org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean 方法里面。这就是 Spring 的源码,而且是一段比较重要的源码。举个简单的例子,比如去调试这部分源码的时候,可以看到这行代码:2021-05-22-16-47-07-445604.png

    Set default singleton scope, if not configured before.

再举个例子:2021-05-22-16-47-07-578090.png这里有三个分支。前两个,一个是作用域为单例的情况、一个是作用域为多例(原型)的情况。一个 bean 的作用域既不是单例、也不是多例,那会是什么?很明显就是自定义作用域了。单例和多例是 Spring 去管理实例的方式而已。Spring 当然也留下了口子,可以按照自己的想法去管理自己的实例。那就是自定义作用域:org.springframework.beans.factory.config.Scope2021-05-22-16-47-07-733191.png这是一个接口,这个接口就是口子。而这个自定义作用域对应的 Spring 源码的入口就是 doGetBean 方法。

以太坊cppgolang区别 编程

以太坊cppgolang区别

以太坊是一种去中心化的开源平台,它采用智能合约技术,旨在构建和运行不受干扰的分布式应用程序。作为目前最受欢迎的区块链平台之一,以太坊提供了多种编程语言的支持,其
progolang 编程

progolang

Go语言(Golang)是由Google开发的一门静态类型编程语言。作为一名专业的Golang开发者,我深知这门语言的优势和特点。在本文中,我将介绍Golang
golangn个发送者 编程

golangn个发送者

Golang是一种开源的编程语言,由Google团队开发,旨在提高程序的并发性和简化软件开发过程。在Go语言中,有时需要向多个接收者发送信息。本文将介绍如何在G
golang技能图谱 编程

golang技能图谱

从互联网行业的快速发展到人工智能技术的日益成熟,各种编程语言也应运而生。而在这众多的编程语言中,Golang(即Go)作为一门强大且高效的开发语言备受关注。Go
评论:0   参与:  0