1、先说明下 Controller默认情况 单例的问题:
使用Spring MVC有一段时间了,之前一直使用Struts2,在struts2中action都是原型(prototype)的, 说是因为线程安全问题,对于Spring MVC中bean默认都是(singleton)单例的,那么用@Controller注解标签注入的Controller类是单例实现的?
测试也表明spring中的controller默认是单例的,若是某个controller中有一个私有的变量i,所有请求到同一个controller时,使用的i变量是共用的,即若是某个请求中修改了这个变量a,则,在别的请求中能够读到这个修改的内容。 若是在@Controller之前增加@Scope(“prototype”),就可以改变单例模式为多例模式
private int cnt = 100;
@RequestMapping("/concurrentRequest")
@ResponseBody
public Map<String ,Object> concurrentRequest()
{
cnt = cnt + 100;
Map result = new HashMap();
String name = Thread.currentThread().getName();
result.put("Success",true);
result.put("Cnt_Value",name + " --> " + cnt);
return result;
}
测试结果如下图:
2、Springmvc :默认singleton 单例模式,Springmvc 是基于方法的,同一个url的请求是同一个实例处理的。每次请求都会把请求参数传递到同一个方法中,方法压栈处理,不会有多线程问题。注意此处是方法处理。此时如果类里面有成员变量,那么这个变量就不是线程安全的了(例如上面的例子 cnt
; 这个变量如果想线程安全则可以用ThreadLocal
)。
以下测试ThreadLocal的线程安全性:
private int cnt = 100;
ThreadLocal<Integer> threadLocalInt = ThreadLocal.withInitial(() -> 0);
@RequestMapping("/concurrentSafeReq")
@ResponseBody
public Map<String,Object> concurrentSafeReq()
{
int temp = threadLocalInt.get();
int resultInt = cnt + temp;
threadLocalInt.set(resultInt);
Map result = new HashMap();
String name = Thread.currentThread().getName();
result.put("Success",true);
result.put("Cnt_Value",name + " --> " + resultInt);
return result;
}
运行结果: