SpringMVC 或 SpringBoot 默认是单例模式(Singleton),多个请求是访问的同一个方法,是如何实现线程安全的?

SpringMVC Controller默认情况下是Singleton(单例)的,当request过来,不用每次创建Controller,会用原来的instance去处理。那么当多个线程调用它的时候,会不会发生线程不安全呢?

1、先说明下 Controller默认情况 单例的问题:

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

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

得到的返回结果如下。

第一次: singletonInt=15

第二次: singletonInt=30

第三次: singletonInt=45

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

 

2、对别Struts与springmvc对比

Struts2          :默认prototype,Struts2 是基于类的,处于线程安全的考虑,采用了prototype模式,也就是说每次请求都会新建一个类来处理,自然就没有线程安全问题了,每次请求的类和数据都是单独的。

Springmvc   :默认singleton 单例模式,Springmvc   是基于方法的,同一个url的请求是同一个实例处理的。每次请求都会把请求参数传递到同一个方法中,此时如果类里面有成员变量,那么这个变量就不是线程安全的了(例如上面的例子  private int singletonInt=1;  这个变量如果想线程安全则可以用ThreadLocal)。

在类中没有成员变量的前提下则是线程安全的。

 

参考:https://zhuanlan.zhihu.com/p/103171449

https://blog.csdn.net/sinat_30474567/article/details/64924927

 

posted @ 2020-04-26 09:12  zhaoyansheng  阅读(5790)  评论(0编辑  收藏  举报