spring 中的 Bean 是否线程安全

结论

其实,Spring 中的 Bean 是否线程安全,其实跟 Spring 容器本身无关Spring框架中没有提供线程安全的策略,因此,spring容器中的Bean本身也不具备线程安全的特性。俺们要透彻理解这个结论,我们首先要知道spring中的Bean是从那里来的

1、spring中的Bean从哪里来?

在 Spring 容器中,除了很多 Spring 内置的 Bean 以外,其他的 Bean 都是我们自己通过 Sring 配置来声明的,然后,由 Spring 容器同意加载。我们在 Spring 声明配置中通常会配置一下内容,如:class(全类名)、id(也就是Bean 的唯一标识)、scope(作用域)以及 lazy-init(是否延时加载)等。之后,Spring容器根据配置内容使用对应的策略来创建 Bean 的实例。因此,Spring 容器中的 Bean 其实都是根据我们自己写的类来创建的实例。因此, Spring 中的 Bean 是否线程安全,跟 Spring 容器无关,只是交由 Spring 容器托管而已
那么,在 Spring 容器中,什么样的 Bean 会存在线程安全问题呢?
回答这个问题之前我们得先回忆一下 Spring Bean 的作用域。在 Spring 定义的作用域中,其中有 prototype(多例 Bean)和 singleton(单例 Bean)。那么,定义为 prototype 的 Bean,是在每次 getBean 的时候都会创建一个新的对象。定义为 singleton 的 Bean,在 Spring 容器中只会存在一个全局共享的实例。基于对以上 Spring Bean 作用域的理解,下面我们来分析一下在 Spring 容器中,什么样的Bean 会存在线程安全问题

2、spring中什么样的Bean存在线程安全问题?

我们已经知道,多例 Bean 每次都会新创建新实例,也就是说线程之间不存在 Bean 共享的问题。因此,多例 Bean 是不存在线程安全问题的。
而单例 Bean 是所有线程共享的一个实例,因此,就可能会存在线程安全问题。但是单例 Bean 又分为无状态 Bean 和有状态 Bean 。在多线程操作中只会对 Bean 的成员变量进行查询操作,不会修改成员变量的值,这样的 Bean称之为无状态 Bean 。所以,可想而知,无状态的单例 Bean 是不存在线程安全问题的但是在多线程操作中如果需要对 Bean 中的成员变量进行数据更新操作,这样的 Bean 称之为有状态 Bean ,所以,有状态的单例 Bean 就可能存在线程安全问题。
所以,最终我们得出结论,在 Spring 中,只有有状态的单例 Bean 才会存在线程安全问题。我们在使用spring的过程中,经常会使用到有状态的单例 Bean,如果真正遇到了线程安全问题,我们该如何处理呢?

3、如何处理spring Bean的线程安全问题?

处理有状态的 Bean 的线程安全问题有以下三种方法:

  1. 将 Bean 的作用域由“singleton”单例改为“prototype”多例
  2. 在 Bean 对象中避免定义可变的成员变量,当然,这样做不太现实,就当我没说
  3. 在类中定义 Thread Local 的成员变量,并将需要的可变成员便来给你保存在 Thread Local 中,Thread Local 本身就具备线程隔离的特性,这就相当于为每个线程提供了一个独立的变量副本,每个线程只需要操作自己的线程副本变量,从而解决线程安全问题
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

胖天才小朱(怀玉)

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值