所谓单例就是所有的请求都用一个对象来处理,比如我们常用的service和dao层的对象通常都是单例的,而多例则指每个请求用一个新的对象来处理,比如action;
讲解
controller默认是单例的,不要使用非静态的成员变量,否则会发生数据逻辑混乱。正因为单例所以不是线程安全的。
下面我们来简单的验证一下:
package com.kafka.demo.controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
*
* @author brb
* @date 2020年09月01日
*
*/
@RestController
public class ScopeTestController {
private int num = 0;
@RequestMapping("/testScope")
public int testScope() {
return ++num;
}
@RequestMapping("/testScope2")
public int testScope2() {
return ++num;
}
}
我们首先访问http://localhost:8080/SpringMvcWithKafka/testScope,得到的答案是1;然后我们再访问http://localhost:8080/SpringMvcWithKafka/testScope2,得到的答案是2。
得到的不同的值,这是线程不安全的。
接下来我们给controller增加作用多例@Scope(“prototype”)
package com.kafka.demo.controller;
import org.springframework.context.annotation.Scope;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
*
* @author brb
* @date 2020年09月01日
*
*/
@RestController
@Scope("prototype")
public class ScopeTestController {
private int num = 0;
@RequestMapping("/testScope")
public int testScope() {
return ++num;
}
@RequestMapping("/testScope2")
public int testScope2() {
return ++num;
}
}
我们依旧先访问http://localhost:8080/SpringMvcWithKafka/testScope,得到的答案是1,我们接着再访问http://localhost:8080/SpringMvcWithKafka/testScope2,得到的答案依旧是1。
相信大家不难发现:
单例是不安全的,会导致属性重复使用。
解决方案
1、不要在controller中定义成员变量。
2、万一必须定义一个非静态成员变量的时候,则通过注解@Scope(“prototype”),将其设置为多例模式。
3、在Controller中使用ThreadLocal变量。
补充说明
spirng bean的作用域有以下五个:
- singleton:单例模式,当spring创建applicationContext容器的时候,spirng会欲初始化所有的该作用域实例,加上lazy-init就可以避免预处理;
- prototype:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spirng将不再对其管理;
- request:搞web的大家都应该明白request的域了吧,就是每次请求都新产生一个实例和prototype不同就是创建后,接下来的管理,spirng依然在监听;
- session:每次会话,同上;
- global session:全局的web域,类似于servlet中的application。