答案:

controller默认是单例的,不要使用非静态的成员变量,否则会发生数据逻辑混乱。正因为单例所以不是线程安全的。

我们下面来简单的验证下:

  1. @Controller
  2. public class ScopeTestController {
  3. private int num = 0;
  4. @RequestMapping("/testScope")
  5. public void testScope() {
  6. System.out.println(++num);
  7. }
  8. @RequestMapping("/testScope2")
  9. public void testScope2() {
  10. System.out.println(++num);
  11. }
  12. }

我们首先访问 http://localhost:8080/testScope,得到的答案是1;然后我们再访问 http://localhost:8080/testScope2,得到的答案是 2。

得到的不同的值,这是线程不安全的。

接下来我们再来给controller增加作用多例 @Scope(“prototype”)

  1. @Controller
  2. @Scope("prototype")
  3. public class ScopeTestController {
  4. private int num = 0;
  5. @RequestMapping("/testScope")
  6. public void testScope() {
  7. System.out.println(++num);
  8. }
  9. @RequestMapping("/testScope2")
  10. public void testScope2() {
  11. System.out.println(++num);
  12. }
  13. }

我们依旧首先访问 http://localhost:8080/testScope,得到的答案是1;然后我们再访问 http://localhost:8080/testScope2,得到的答案还是 1。

相信大家不难发现 :

单例是不安全的,会导致属性重复使用。

1. 解决方案

  1. 不要在controller中定义成员变量。
  2. 万一必须要定义一个非静态成员变量时候,则通过注解@Scope(“prototype”),将其设置为多例模式。
  3. 在Controller中使用ThreadLocal变量

2. 补充说明

spring bean作用域有以下5个:

  • singleton:单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作用域实例,加上lazy-init就可以避免预处理;
  • prototype:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理;
    (下面是在web项目下才用到的)
  • request:搞web的大家都应该明白request的域了吧,就是每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;
  • session:每次会话,同上;
  • global session:全局的web域,类似于servlet中的application。