spring bean单例和并发

本文探讨了Spring框架中bean的默认单例模式及如何确保线程安全,特别是通过ThreadLocal实现。同时对比了Spring单例与Java单例的区别。

spring中的bean默认都是单例的,如下所示,效果都是一样:

<bean name="" class=""/>

<bean name="" class="" scope="singleton"/>

<bean name="" class="" singleton="true"/>


所以我们的controller,service都是单例的,这些一般都是属于我们应用中的业务对象,spring的单例不会影响到应用的并发访问,这些业务对象没有做线程的并发限制,因此在使用这些业务对象的时候,一定要声明一些成员变量(即将变量放到方法中),如果不放在方法中,而是声明成成员变量,就会出现并发问题,尤其对于可变的状态变量,例如:

声明成员变量如下:int i=0;

在方法中有对i进行修改的地方,这个时候,多线程访问就会出现并发问题,所以不要在业务对象中声明状态可变化的成员变量。


那么spring是怎么解决非线程安全的呢?例如,对于事物管理transactionSynchronizationManager这种

spring是通过ThradLocal来处理的,ThradLocal相当于一个成员变量,是线程安全的成员变量,每一个线程都保存一份,具体,可以查看ThradLocal的源码,说的很清楚,另外通过ThreadLocal比synchronized效率高,ThreadLocal是以空间换时间,synchronized是以时间换空间。


最后谈谈spring的单例跟java的单例的区别:

spring的单例是对于IOC容器来说的。

java的单例是针对整个JVM来说的,ClassLoader内的都共享此实例。


Spring中,Bean的默认作用域是模式,即每个容器只会创建一个实。对于模式的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、付费专栏及课程。

余额充值