spring中的controller默认是单例的,不要使用非静态的成员变量,否则会发生并发安全性问题。
@RestController
public class ScopeController {
private int num = 0;
@GetMapping("/testScope")
public Object testScope() {
System.out.println(++num);
return num;
}
@GetMapping("/testScope2")
public Object testScope2() {
System.out.println(++num);
return num;
}
}
1、首先访问testScope,得到返回1;
2、然后访问testScope2,得到返回2;此时不是想要的结果,num被修改了。
因为controller是单例不安全的,所有request请求都访问同一个controller时,成员变量是公用的,如果某个请求修改了这个变量,那么在其他的请求中可以获取修改过的变量。
解决方案
1、不要在controller中定义成员变量;
2、若必须定义一个非静态成员变量时,通过注解@Scope(“prototype”)将其设置为多例模式,则每次请求都创建新的controller;
测试结果:每次请求获得的返回都是1。
3、使用ThreadLocal变量;
测试结果:连续请求得到的返回有1、2、3等,可以看到服务器默认线程池大小为10,这10个可以被request请求复用,所以会出现如上的结果;
@RestController
public class ScopeController {
private ThreadLocal<Integer> num = new ThreadLocal<>();
@GetMapping("/testScope")
public Object testScope() {
if (num.get() == null) {
num.set(0);
}
num.set(num.get().intValue() + 1);
return num.get() + Thread.currentThread().getName();
}
@GetMapping("/testScope2")
public Object testScope2() {
if (num.get() == null) {
num.set(0);
}
num.set(num.get().intValue() + 1);
return num.get() + Thread.currentThread().getName();
}
}
ThreadLocal可以达到线程隔离,但还是无法达到并发安全。
4、使用并发安全类;
spring bean的作用域如下;
- singleton:单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作用域实例;
- prototype:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理;
- request:每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;
- session,每次会话都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;
- global session:全局的web域,类似于servlet中的application;