spring 的 controller 是单例模式,但是是多线程,各个线程之间不影响

简介: spring 的 controller 是单例模式,但是是多线程,各个线程之间不影响

spring mvc 的 Controller 类默认 Scope 是单例 (singleton) 的

使用 Spring MVC 有一段时间了,之前一直使用 Struts2, 在 struts2 中 action 都是原型 (prototype) 的, 说是因为线程安全问题,对于 Spring MVC 中 bean 默认都是 (singleton) 单例的,那么用 @Controller 注解标签注入的 Controller 类是单例实现的?

 

测试结果发现 spring3 中的 controller 默认是单例的,若是某个 controller 中有一个私有的变量 i, 所有请求到同一个 controller 时,使用的 i 变量是共用的,即若是某个请求中修改了这个变量 a,则,在别的请求中能够读到这个修改的内容。 若是在 @Controller 之前增加 @Scope (“prototype”),就可以改变单例模式为多例模式

以下是测试步骤,代码与结果.

1. 如果是单例类型类的,那么在 Controller 类中的类变量应该是共享的,如果不共享,就说明 Controller 类不是单例。以下是测试代码:

 

  1. import javax.servlet.http.HttpServletRequest;
  2. import javax.servlet.http.HttpServletResponse;
  3.  
  4. import org.springframework.stereotype.Controller;
  5. import org.springframework.web.bind.annotation.RequestMapping;
  6. import org.springframework.web.bind.annotation.ResponseBody;
  7. @Controller
  8. public class ExampleAction {
  9. private int singletonInt=0;
  10. @RequestMapping(value = "/test")
  11. @ResponseBody
  12. public String singleton(HttpServletRequest request,
  13. HttpServletResponse response) throws Exception {
  14. String data=request.getParameter("data");
  15. if(data!=null&&data.length()>0){
  16. try{
  17. int paramInt= Integer.parseInt(data);
  18. singletonInt = singletonInt + paramInt;
  19. }
  20. catch(Exception ex){
  21. singletonInt+=10;
  22. }
  23. }else{
  24. singletonInt+=1000;
  25. }
  26. return String.valueOf(singletonInt);
  27. }
  28. }

 

分别三次请求: http://localhost:8080/example/test.do?data=15

得到的返回结果如下。

第一次: singletonInt=15

第二次: singletonInt=30

第三次: singletonInt=45

从以上结果可以得知,singletonInt 的状态是共享的,因此 Controller 是单例的。

2. 如果 Controller 类是单例,那么多个线程请求同一个 Controller 类中的同一个方法,线程是否会堵塞

 

 

  1. @RequestMapping(value = "/sleepdata")
  2. @ResponseBody
  3. public String switcher(HttpServletRequest request
  4. , HttpServletResponse response)
  5. throws Exception {
  6. String sleep = request.getParameter("sleep");
  7. if (sleep.equals("on")) {
  8. Thread.currentThread().sleep(100000);
  9. return "sleep on";
  10. } else {
  11. return sleep;
  12. }
  13. }

 

验证方法:分别发送两个请求,

第一个请求:http://localhost:8080/coreplat/sleepdata.do?sleep=on

第二个请求:http://localhost:8080/coreplat/sleepdata.do?sleep=test

验证结果:第一个请求发出去以后,本地服务器等待 100s,然后返回结果”sleep on”, 在本地服务器等待的者 100s 当中,发送第二个请求,直接返回结果”test”。说明之间的线程是不互相影响的。

相关文章
|
2天前
|
安全 Java 关系型数据库
多线程(线程安全)
多线程(线程安全)
12 4
|
2天前
|
Java 调度
多线程(创建多线程的五种方式,线程状态, 线程中断)
多线程(创建多线程的五种方式,线程状态, 线程中断)
12 0
|
2天前
|
Linux 调度 Windows
【操作系统】线程、多线程模型
【操作系统】线程、多线程模型
11 1
|
2天前
|
Java 调度
多线程的基本概念和实现方式,线程的调度,守护线程、礼让线程、插入线程
多线程的基本概念和实现方式,线程的调度,守护线程、礼让线程、插入线程
11 0
|
3天前
|
设计模式 安全 Java
Java多线程案例-Java多线程(3)
Java多线程案例-Java多线程(3)
8 1
|
3天前
|
存储 安全 Java
Java多线程安全风险-Java多线程(2)
Java多线程安全风险-Java多线程(2)
9 1
|
3天前
|
Java 调度
聊聊Java线程是个啥东西-Java多线程(1)
聊聊Java线程是个啥东西-Java多线程(1)
10 0
|
3天前
|
NoSQL Java 关系型数据库
实时计算 Flink版产品使用合集之实现存量读取时采用多线程、增量读取时采用单线程如何解决
实时计算Flink版作为一种强大的流处理和批处理统一的计算框架,广泛应用于各种需要实时数据处理和分析的场景。实时计算Flink版通常结合SQL接口、DataStream API、以及与上下游数据源和存储系统的丰富连接器,提供了一套全面的解决方案,以应对各种实时计算需求。其低延迟、高吞吐、容错性强的特点,使其成为众多企业和组织实时数据处理首选的技术平台。以下是实时计算Flink版的一些典型使用合集。
|
4天前
|
缓存 NoSQL Redis
【后端面经】【缓存】36|Redis 单线程:为什么 Redis 用单线程而 Memcached 用多线程?-- Redis多线程
【5月更文挑战第21天】Redis启用多线程后,主线程负责接收事件和命令执行,IO线程处理读写数据。请求处理流程中,主线程接收客户端请求,IO线程读取并解析命令,主线程执行后写回响应。业界普遍认为,除非必要,否则不建议启用多线程模式,因单线程性能已能满足多数需求。公司实际场景中,启用多线程使QPS提升约50%,或选择使用Redis Cluster以提升性能和可用性。
9 0
|
4天前
|
设计模式 安全 Java
【Linux 系统】多线程(生产者消费者模型、线程池、STL+智能指针与线程安全、读者写者问题)-- 详解
【Linux 系统】多线程(生产者消费者模型、线程池、STL+智能指针与线程安全、读者写者问题)-- 详解