初步理解Spring中Bean的单例模式

自学完SpringMVC之后,就产生了对于Spring中Bean的误解,此前我的理解是:单例模式,那每个controller同一时间不就只能处理一个请求。直到最近重新学习了JUC相关的内容,才搞明白单例模式不会影响多线程的调用,只是确保此类只有一个实例对象,明白这个之后才知道自己是被多么简单的问题给迷惑了。
接下来是我通过一个小demo,固化自己的理解。
这里懒得新建项目,就只在现有项目里添了几个测试方法。

通过当前controller(itemController)调用signalItem输出相关基本信息,并暂停10s

同时subscribeController调用signalSub输出线程名及itemController的地址,并且同时调用itemController的signalTest方法确保itemController可以正常调用
image-20220813224702107
image-20220813225218880
先向signalItem发请求,然后向signalSub发请求,可以得到如下结果:
image-20220813225728782
通过结果不难看出,每次请求进来都会由一个线程来处理(这些线程通过tomcat的线程池维护)。并且两个请求的itemController是同一个对象,也就是说Spring的Bean是单例的。在controller对象没有处理完当前请求时,可以在另一个线程里处理其它请求,因为我们并没有对这个controller加锁。根本不会发生不能同时处理多个请求的情况。

SpringBean的默认作用域是单例模式,即每个容器只会创建一个实例。对于单例模式Bean,如果多个线程同时访问该Bean的方法或属性,可能会引发线程安全问题。这是因为多个线程共享同一个实例,对实例的修改可能会相互干扰。因此,需要特别注意在单例模式下处理线程安全问题。 对于解决单例模式下的线程安全问题,可以采取以下几种方式: 1. 方法级别加锁:在需要保证线程安全的方法上使用synchronized关键字,确保同一时间只能有一个线程访问该方法。这种方式简单直接,但会降低并发性能。 2. 使用ThreadLocal:ThreadLocal是Java提供的一种线程级别的变量隔离机制。可以将需要共享的对象存储在ThreadLocal,每个线程都拥有自己的副本,避免了线程安全问题。可以将非线程安全的对象存储在ThreadLocal,确保每个线程都使用自己的对象副本。 3. 将Bean的作用域设置为prototype:使用prototype作用域的Bean在每次被注入或获取时都会创建一个新的实例,解决了单例模式下的线程安全问题。可以通过在Bean的配置文件设置scope属性为"prototype"来实现。 总结起来,为了解决Spring Bean单例模式下的线程安全问题,可以采用方法级别加锁、使用ThreadLocal或将Bean的作用域设置为prototype等方式来保证线程安全性。具体选择哪种方式需要根据实际场景和需求来决定。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值