Spring 的 Controller 是单例还是多例?怎么保证并发的安全

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);     } }

 

我们首先访问 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; /**  * @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);     } }

 

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

 

相信大家不难发现 :

 

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

推荐一个DD写的SpringBoot基础教程:http://blog.didispace.com/spring-boot-learning-2x/

1|0解决方案

 

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

 

2|0补充说明

 

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。


__EOF__

本文作者菜菜
本文链接https://www.cnblogs.com/caicz/p/14923276.html
关于博主:评论和私信会在第一时间回复。或者直接私信我。
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。您的鼓励是博主的最大动力!
posted @   菜菜聊架构  阅读(100)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
阅读排行:
· winform 绘制太阳,地球,月球 运作规律
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
历史上的今天:
2019-06-23 Kafka主题体系架构-复制、故障转移和并行处理
2019-06-23 Kafka如何实现每秒上百万的高并发写入
点击右上角即可分享
微信分享提示