当前位置:首页 > 科技  > 软件

面试官:单例Bean一定不安全吗?实际工作中如何处理此问题?

来源: 责编: 时间:2024-01-15 09:21:45 273观看
导读默认情况下,Spring Boot 中的 Bean 是非线程安全的。这是因为,默认情况下 Bean 的作用域是单例模式,那么此时,所有的请求都会共享同一个 Bean 实例,这意味着这个 Bean 实例,在多线程下可能被同时修改,那么此时它就会出现线程

wxA28资讯网——每日最新资讯28at.com

默认情况下,Spring Boot 中的 Bean 是非线程安全的。这是因为,默认情况下 Bean 的作用域是单例模式,那么此时,所有的请求都会共享同一个 Bean 实例,这意味着这个 Bean 实例,在多线程下可能被同时修改,那么此时它就会出现线程安全问题。wxA28资讯网——每日最新资讯28at.com

wxA28资讯网——每日最新资讯28at.com

Bean 的作用域(Scope)指的是确定在应用程序中创建和管理 Bean 实例的范围。也就是在 Spring 中,可以通过指定不同的作用域来控制 Bean 实例的生命周期和可见性。例如,单例模式就是所有线程可见并共享的,而原型模式则是每次请求都创建一个新的原型对象。wxA28资讯网——每日最新资讯28at.com

wxA28资讯网——每日最新资讯28at.com

1、单例Bean一定不安全吗?

并不是,单例 Bean 分为以下两种类型:wxA28资讯网——每日最新资讯28at.com

  • 无状态 Bean(线程安全):Bean 没有成员变量,或多线程只会对 Bean 成员变量进行查询操作,不会修改操作。
  • 有状态 Bean(非线程安全):Bean 有成员变量,并且并发线程会对成员变量进行修改操作。

所以说:有状态的单例 Bean 是非线程安全的,而无状态的 Bean 是线程安全的wxA28资讯网——每日最新资讯28at.com

wxA28资讯网——每日最新资讯28at.com

但在程序中,只要有一种情况会出现线程安全问题,那么它的整体就是非线程安全的,所以总的来说,单例 Bean 还是非线程安全的。wxA28资讯网——每日最新资讯28at.com

wxA28资讯网——每日最新资讯28at.com

(1)无状态的Bean

无状态的 Bean 指的是不存在成员变量,或只有查询操作,没有修改操作,它的实现示例代码如下:wxA28资讯网——每日最新资讯28at.com

import org.springframework.stereotype.Service;@Servicepublic class StatelessService {    public void doSomeTask() {        // 执行任务    }}

(2)有状态的Bean

有成员变量,并且存在对成员变量的修改操作,如下代码所示:wxA28资讯网——每日最新资讯28at.com

import org.springframework.stereotype.Service;@Servicepublic class UserService {    private int count = 0;    public void incrementCount() {        count++; // 非原子操作,并发存在线程安全问题    }    public int getCount() {        return count;    }}

2、如何保证线程安全?

想要保证有状态 Bean 的线程安全,可以从以下几个方面来实现:wxA28资讯网——每日最新资讯28at.com

  • 使用 ThreadLocal(线程本地变量):每个线程修改自己的变量,就没有线程安全问题了。
  • 使用锁机制:例如 synchronized 或 ReentrantLock 加锁修改操作,保证线程安全。
  • 设置 Bean 为原型作用域(Prototype):将 Bean 的作用域设置为原型,这意味着每次请求该 Bean 时都会创建一个新的实例,这样可以防止不同线程之间的数据冲突,不过这种方法增加了内存消耗。
  • 使用线程安全容器:例如使用 Atomic 家族下的类(如 AtomicInteger)来保证线程安全,此实现方式的本质还是通过锁机制来保证线程安全的,Atomic 家族底层是通过乐观锁 CAS(Compare And Swap,比较并替换)来保证线程安全的。

具体实现如下。wxA28资讯网——每日最新资讯28at.com

(1)使用ThreadLocal保证线程安全

实现代码如下:wxA28资讯网——每日最新资讯28at.com

import org.springframework.stereotype.Service;@Servicepublic class UserService {    private ThreadLocal<Integer> count = ThreadLocal.withInitial(() -> 0);    public void incrementCount() {        count.set(count.get() + 1);    }    public int getCount() {        return count.get();    }}

使用 ThreadLocal 需要注意一个问题,在用完之后记得调用 ThreadLocal 的 remove 方法,不然会发生内存泄漏问题。wxA28资讯网——每日最新资讯28at.com

(2)使用锁机制

锁机制中最简单的是使用 synchronized 修饰方法,让多线程执行此方法时排队执行,这样就不会有线程安全问题了,如下代码所示:wxA28资讯网——每日最新资讯28at.com

import org.springframework.stereotype.Service;@Servicepublic class UserService {    private int count = 0;    public synchronized void incrementCount() {        count++; // 非原子操作,并发存在线程安全问题    }    public int getCount() {        return count;    }}

(3)设置为原型作用域

原型作用域通过 @Scope("prototype") 来设置,表示每次请求时都会生成一个新对象(也就没有线程安全问题了),如下代码所示:wxA28资讯网——每日最新资讯28at.com

import org.springframework.stereotype.Service;@Service@Scope("prototype")public class UserService {    private int count = 0;    public void incrementCount() {        count++; // 非原子操作,并发存在线程安全问题    }    public int getCount() {        return count;    }}

(4)使用线程安全容器

我们可以使用线程安全的容器,例如 AtomicInteger 来替代 int,从而保证线程安全,如下代码所示:wxA28资讯网——每日最新资讯28at.com

import org.springframework.stereotype.Service;import java.util.concurrent.atomic.AtomicInteger;@Servicepublic class UserService {    private AtomicInteger count = new AtomicInteger(0);    public void incrementCount() {        count.incrementAndGet();    }    public int getCount() {        return count.get();    }}

实际工作中如何保证线程安全?

实际工作中,通常会根据具体的业务场景来选择合适的线程安全方案,但是以上解决线程安全的方案中,ThreadLocal 和原型作用域会使用更多的资源,占用更多的空间来保证线程安全,所以在使用时通常不会作为最佳考虑方案。wxA28资讯网——每日最新资讯28at.com

而锁机制和线程安全的容器通常会优先考虑,但需要注意的是 AtomicInteger 底层是乐观锁 CAS 实现的,因此它存在乐观锁的典型问题 ABA 问题(如果有状态的 Bean 中既有 ++ 操作,又有 -- 操作时,可能会出现 ABA 问题),此时就要使用锁机制,或 AtomicStampedReference 来解决 ABA 问题了。wxA28资讯网——每日最新资讯28at.com

小结

单例模式的 Bean 并不一定都是非线程安全的,其中有状态的 Bean 是存在线程安全问题的。实际工作中通常会使用锁机制(synchronized 或 ReentrantLock)或线程安全的容器来解决 Bean 的线程安全问题,但具体使用哪种方案,还要结合具体业务场景来定。wxA28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-60980-0.html面试官:单例Bean一定不安全吗?实际工作中如何处理此问题?

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: Go语言常见错误—Any 没传递任何信息

下一篇: 使用Java 17中的 record 替代 Lombok 的部分功能

标签:
  • 热门焦点
  • 红魔电竞平板评测:大屏幕硬实力

    前言:三年的疫情因为要上网课的原因激活了平板市场,如今网课的时代已经过去,大家的生活都恢复到了正轨,这也就意味着,真正考验平板电脑生存的环境来了。也就是面对着这种残酷的
  • 5月iOS设备好评榜:iPhone 14仅排第43?

    来到新的一月,安兔兔的各个榜单又重新汇总了数据,像安卓阵营的榜单都有着比较大的变动,不过iOS由于设备的更新换代并没有那么快,所以相对来说变化并不大,特别是iOS好评榜,老款设
  • 把LangChain跑起来的三个方法

    使用LangChain开发LLM应用时,需要机器进行GLM部署,好多同学第一步就被劝退了,那么如何绕过这个步骤先学习LLM模型的应用,对Langchain进行快速上手?本片讲解3个把LangChain跑起来
  • 如何正确使用:Has和:Nth-Last-Child

    我们可以用CSS检查,以了解一组元素的数量是否小于或等于一个数字。例如,一个拥有三个或更多子项的grid。你可能会想,为什么需要这样做呢?在某些情况下,一个组件或一个布局可能会
  • 每天一道面试题-CPU伪共享

    前言:了不起:又到了每天一到面试题的时候了!学弟,最近学习的怎么样啊 了不起学弟:最近学习的还不错,每天都在学习,每天都在进步! 了不起:那你最近学习的什么呢? 了不起学弟:最近在学习C
  • 电视息屏休眠仍有网络上传 爱奇艺被质疑“薅消费者羊毛”

    记者丨宁晓敏 见习生丨汗青出品丨鳌头财经(theSankei) 前不久,爱奇艺发布了一份亮眼的一季报,不仅营收和会员营收创造历史最佳表现,其运营利润也连续6个月实现增长。自去年年初
  • 微博大门常打开,迎接海外画师漂洋东渡

    作者:互联网那些事&ldquo;起猛了,我能看得懂日语了&rdquo;。&ldquo;为什么日本人说话我能听懂?&rdquo;&ldquo;中文不像中文,日语不像日语,但是我竟然看懂了&rdquo;&hellip;&hell
  • 回归OPPO两年,一加赢了销量,输了品牌

    成为OPPO旗下主打性能的先锋品牌后,一加屡创佳绩。今年618期间,一加手机全渠道销量同比增长362%,凭借一加 11、一加 Ace 2、一加 Ace 2V三款爆品,一加
  • 华为举行春季智慧办公新品发布会 首次推出电子墨水屏平板

    北京时间2月27日晚,华为在巴塞罗那举行春季智慧办公新品发布会,在海外市场推出之前已经在中国市场上市的笔记本、平板、激光打印机等办公产品,并首次推出搭载
Top