Spring 中的bean 默认情况下是单例模式 不安全
先了解下bean(scope)的作用域
1、singleton:单例,默认作用域。
2、prototype:原型,每次创建一个新对象。
3、request:请求,每次Http请求创建一个新对象,适用于WebApplicationContext环境下。
4、session:会话,同一个会话共享一个实例,不同会话使用不用的实例。
5、global-session:全局会话,所有会话共享一个实例。
此次只讨论单例模式,测试如下
package com.example.demo.kang;
import org.springframework.context.annotation.Scope;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping(value = "kangtest")
@Scope(value = "singleton")
public class TestController {
private int a = 1;
@RequestMapping(value = "test", method = RequestMethod.GET)
public int test() {
a = a+1;
System.out.println("a="+a);
return a;
}
}
单例模式下用postman请求3次结果
说明其线程不安全,每次请求a的值都改变了,但我们换成作用域为prototype
不管请求多少次,值都未变,线程安全
原因:单例模式下,spring bean只会在容器生成一个对象,如果这个对象有状态性(即有存储属性),则全局共享,每个线程都会对属性做出改变,不安全。 但是在原型模式下,每次请求都会单独生成一个controller对象,对象之间互不干扰,线程安全。
这里主要在于bean对象是有状态性还是无状态性。平时我们写的bean对象都是无状态性的bean,例如
这里service只是调用,不具备存储功能,所以此时的单例bean是线程安全的。
当然你如果在原型模式下用静态变量必然也是不安全的,原因就不用我说了
那如果我真的想在单例模式下加上变量,又想线程安全怎么做呢? 用ThreadLocal
package com.example.demo.kang;
import org.springframework.context.annotation.Scope;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping(value = "kangtest")
@Scope(value = "singleton")
public class TestController {
private int a = 1;
ThreadLocal local = new ThreadLocal<>();
@RequestMapping(value = "test", method = RequestMethod.GET)
public int test() {
local.set(a);
System.out.println("a="+a);
return a;
}
}
用postman测试,无论多少次都是一样



