​阅读大约需要2分钟​

答案:


controller默认是单例的,不要使用非静态的成员变量,否则会发生数据逻辑混乱。

正因为单例所以不是线程安全的。


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

package com.riemann.springbootdemo.controller;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

/**
* @author riemann
* @date 2019/07/29 22:56
*/
@Controller
public class ScopeTestController {

private int num = 0;

@RequestMapping("/testScope")
public void testScope() {
System.out.println(++num);
}

@RequestMapping("/testScope2")
public void testScope2() {
System.out.println(++num);
}

}

我们首先访问 ​​​​,得到的答案是 ​​1​​;

然后我们再访问 ​​​​,得到的答案是 ​​2​​。


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



接下来我们再来给 Controller 增加作用多例 ​​@Scope("prototype")​

package com.riemann.springbootdemo.controller;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

/**
* @author riemann
* @date 2019/07/29 22:56
*/
@Controller
@Scope("prototype")
public class ScopeTestController {

private int num = 0;

@RequestMapping("/testScope")
public void testScope() {
System.out.println(++num);
}

@RequestMapping("/testScope2")
public void testScope2() {
System.out.println(++num);
}

}

我们依旧首先访问 ​​​​,得到的答案是 ​​1​​;

然后我们再访问 ​​​​,得到的答案还是 ​​1​​。


相信大家不难发现 :


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


解决方案


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


补充说明

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。


2021 JavaPub版 面试题全力创作中,推荐给你做好的朋友(搜:JavaPub版)