首先明确:Spring的controller默认是单例的,不要使用非静态的成员变量,否则会发生数据逻辑混乱。
其次要明白:因为是单例模式,所以是非线程安全的。
下面用简单代码示例下:
package com.riemann.springbootdemo.controller;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@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);
}
}
我们首先访问 http://localhost:8080/testScope,得到的答案是1;
然后我们再访问 http://localhost:8080/testScope2,得到的答案是 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;
@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);
}
}
我们依旧首先访问 http://localhost:8080/testScope,得到的答案是1;
然后我们再访问 http://localhost:8080/testScope2,得到的答案还是 1。
这个时候是controller变成多例模式了,是线程安全的。
解决方案:
1、不要在controller中定义成员变量。
2、万一必须要定义一个非静态成员变量时候,则通过注解@Scope("prototype"),将controller设置为多例模式。
3、在Controller中使用ThreadLocal变量。
补充说明:
spring bean作用域有以下5个:
singleton: 单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作用域实例,加上lazy-init就可以避免预处理;
prototype:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理;
(下面是在web项目下才用到的)
request:每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;
session: 每次会话,同request;
global session: 全局的web域,类似于servlet中的application。
本文探讨了Spring Controller默认的单例模式导致的数据逻辑混乱问题,强调了线程不安全的特性。通过代码示例展示了如何通过@Scope(prototype)注解切换到多例模式,确保线程安全。同时提出了避免在Controller中定义成员变量,使用ThreadLocal等解决方案。总结了Spring Bean的五种作用域,并提到了在Web项目中的request、session和globalsession作用域。
1613

被折叠的 条评论
为什么被折叠?



