Java 多线程进阶:什么是线程安全?

在多线程编程中,“线程安全”是一个非常重要但又常被误解的概念。尤其对于刚接触多线程的人来说,不理解线程安全的本质,容易写出“偶尔出错”的代码——这类 bug 往往隐蔽且难以复现。

本文将用尽可能通俗的语言,从三个角度解释线程不安全的常见原因,并提供具体的示例和解决方法。


一、什么是线程安全?

线程安全(Thread Safety)简单来说就是:

在多个线程同时执行某段代码时,无论线程怎么调度、怎么交叉执行,都会得到正确的结果,不会出 bug。

线程安全:

多个线程访问相同对象时,不会引起数据不一致或状态混乱。

线程不安全:

同一段代码,在单线程环境下一切正常,但在多线程环境下,结果可能出错或不一致。


二、线程不安全的三大根源

1. 原子性问题:操作不是一步完成的

一个典型例子是变量自增 count++。虽然看起来是一条语句,但其实底层是三条指令:

load   // 从内存读取 count 到 CPU 寄存器
add    // 在寄存器中执行 +1 操作
store  // 把结果写回内存

在两个线程同时执行 count++ 时,可能会出现以下竞态(race condition):

🎯 理想顺序(无竞态):线程串行执行

时间轴 →
Thread A:load(0) → add(1) → store(1)
Thread B:  load(1) → add(1) → store(2)

最终结果:count = 2(正确,无操作丢失)


❌ 竞态情况 1:两个线程都读取了旧值(0)

时间轴 →
Thread A:load(0) → add(1) -----------------> store(1)
Thread B:  load(0) → add(1) -----------------> store(1)

最终结果:count = 1 ❌(两个线程都基于旧值 0 进行计算,结果被覆盖,丢失了一次加法)


❌ 竞态情况 2:交错执行导致结果被覆盖

时间轴 →
Thread A:load(0) → add(1) -----------------> store(1)
Thread B:  load(0) → add(1) → store(1)

最终结果:count = 1 ❌(Thread A 的存储操作覆盖了 Thread B 的结果)


❌ 竞态情况 3:Thread B 插队执行完毕

时间轴 →
Thread A:load(0) → add(1)
Thread B:  load(0) → add(1) → store(1)
Thread A:  store(1)

最终结果:count = 1 ❌(Thread A 最后写入的值覆盖了 Thread B 的加法结果)


❌ 竞态情况 4:Thread A 读值后等待,Thread B 先完成

时间轴 →
Thread A:load(0)
Thread B:  load(0) → add(1) → store(1)
Thread A:  add(1) → store(1)

最终结果:count = 1 ❌(两个线程都基于同一个初始值 0 进行加法,导致一次加法丢失)

解决方案

  • 使用 synchronized 同步关键代码块

    synchronized (this) {count++;
    }
    
  • 使用原子类如 AtomicInteger 实现原子操作

    AtomicInteger count = new AtomicInteger(0);
    count.incrementAndGet(); // 原子性 +1
    

    3. 指令重排序:执行顺序被优化

    为了提升性能,编译器和 CPU 可能对指令重新排序,只要单线程语义不变即可。但这可能影响多线程环境的执行逻辑。

    例如:

    // 线程 A
    a = 1;
    flag = true;// 线程 B
    if (flag) {System.out.println(a); // 可能输出 0!
    }
    

     


2. 可见性问题:变量更新对其他线程不可见

Java 中每个线程都有自己的工作内存(工作缓存),它会缓存主内存中的变量副本。这就导致:

  • 一个线程修改了变量,另一个线程却看不到。

例如:

volatile boolean running = true;public void stop() {running = false;
}public void run() {while (running) {// 执行某些操作}
}

由于重排序,可能发生 flag = true 提前执行,而 a = 1 还没发生,导致 a 为默认值 0

🛠 解决方案:

  • 使用 volatile 修饰 flag,防止指令重排;

  • 或使用 synchronized,保证顺序一致;

  • 对不可变对象,使用 final 修饰字段也是一种有效方式。


三、小结

线程安全问题,来源于我们“看似简单”的代码在多线程环境下可能出现的非预期行为:

  • 原子性:多步操作被打断

  • 可见性:线程看不到最新值

  • 指令重排:操作顺序被调整

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/79134.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MSO-Player:基于vlc的Unity直播流播放器,支持主流RTSP、RTMP、HTTP等常见格式

MSO-Player 基于libVLC的Unity视频播放解决方案 支持2D视频和360度全景视频播放的Unity插件 📑 目录 🎥 MSO-Player 📋 功能概述🚀 快速入门📚 关键组件📝 使用案例🔌 依赖项📋 注意…

navicat中导出数据表结构并在word更改为三线表(适用于navicat导不出doc)

SELECTCOLUMN_NAME 列名,COLUMN_TYPE 数据类型,DATA_TYPE 字段类型,IS_NULLABLE 是否为空,COLUMN_DEFAULT 默认值,COLUMN_COMMENT 备注 FROMINFORMATION_SCHEMA.COLUMNS WHEREtable_schema db_animal(数据库名) AND table_name activity(…

docker学习笔记6-安装wordpress

一、创建自定义网络、查看网络 docker netword create blog docker network ls 二、 启动mysql容器 启动命令: docker run -d -p 3306:3306 \ -e MYSQL_ROOT_PASSWORD123456 \ -e MYSQL_DATABASEwordpress \ -v mysql-data:/var/lib/mysql \ -v /app/myconf:/etc…

03_Mybatis-Plus LambadaQueryWrapper 表达式爆空指针异常

&#x1f31f; 03_MyBatis-Plus LambdaQueryWrapper 爆出空指针异常的坑点分析 ❓ 场景描述 来看一段常见的 MyBatis-Plus 查询写法&#xff0c;是否存在问题&#xff1f; Page<VideoInfoVo> videoInfoVosPage videoMapper.selectPage(page, new LambdaQueryWrapper&…

WEB安全--社会工程--SET钓鱼网站

1、选择要钓鱼的网站 2、打开kali中的set 3、启动后依次选择&#xff1a; 4、输入钓鱼主机的地址&#xff08;kali&#xff09;和要伪装的网站域名&#xff1a; 5、投放钓鱼网页&#xff08;服务器域名:80&#xff09; 6、获取账号密码

Ethan独立开发产品日报 | 2025-04-29

1. mrge 代码审查的光标 mrge 是一个由人工智能驱动的代码审查平台&#xff0c;能够自动审核拉取请求&#xff08;PR&#xff09;&#xff0c;为人工审查员提供超级能力。它是像 cal.com 和 n8n 这样快速发展的团队的首选工具。 关键词&#xff1a;mrge, 代码审查, AI驱动, …

ubuntu22.04 qemu arm64 环境搭建

目录 创建 安装 Qemu 启动 # 进入qemu虚拟机后执行 qemu编译器安装 创建 qemu-img create ubuntu22.04_arm64.img 40G 安装 qemu-system-aarch64 -m 4096 -cpu cortex-a57 -smp 4 -M virt -bios QEMU_EFI.fd -nographic -drive ifnone,fileubuntu-22.04.5-live-server-a…

安全生产知识竞赛宣传口号160句

1. 安全生产是责任&#xff0c;每个人都有责任 2. 安全生产是保障&#xff0c;让我们远离危险 3. 安全生产是团结&#xff0c;共同守护每一天 4. 注重安全&#xff0c;守护明天 5. 安全生产无小事&#xff0c;关乎千家万户 6. 安全第一&#xff0c;人人有责 7. 安全生产无差别&…

Python 虚拟环境管理:venv 与 conda 的选择与配置

文章目录 前言一、虚拟环境的核心价值1.1 依赖冲突的典型场景1.2 隔离机制实现原理 二、venv 与 conda 的架构对比2.1 工具定位差异2.2 性能基准测试&#xff08;以创建环境 安装 numpy 为例&#xff09; 三、venv 的配置与最佳实践3.1 基础工作流3.2 多版本 Python 管理 四、…

【自然语言处理与大模型】如何获取特定领域的微调数据集?

在特定领域中&#xff0c;数据集通常由提出需求的一方提供。然而&#xff0c;在某些情况下&#xff0c;如果他们未能提供所需的数据&#xff0c;或者你正在独立开展一个项目&#xff0c;并且需要相应的数据来推进工作&#xff0c;这时你应该怎么办呢&#xff1f;本文提供一种思…

Map系列之ConcurrentHashMap源码分析:高并发场景下的性能密码

引言&#xff1a;当线程安全成为刚需 1.1 并发时代的Map困境 经典案例&#xff1a;电商秒杀系统超卖事故分析&#xff08;附线程堆栈截图&#xff09;传统方案缺陷&#xff1a;synchronizedMap的吞吐量陷阱&#xff08;JMH测试数据对比&#xff09;ConcurrentHashMap的定位&a…

URP - 序列图动画的实现

效果&#xff1a; 【太妃糖耶】更新了一条视频&#xff0c;快来围观&#xff01; 序列图动画的实现 首先先了解下序列图样式的纹理图片 如上图一可在Shader中使用该图片制作燃烧的火的动画&#xff0c;但是如何实现呢&#xff1f;接下来一起来看一下吧 序列图动画的实现原理大…

python中 str.strip() 是什么意思

在 Python 中&#xff0c;str.strip() 是字符串&#xff08;str&#xff09;类型的一个方法&#xff0c;用于移除字符串两端的空白字符&#xff08;默认情况下&#xff09;或指定字符&#xff0c;并返回处理后的新字符串。 语法&#xff1a; str.strip([chars])chars&#xf…

记录idea可以运行但是maven install打包却找不到问题

解决idea使⽤maven多模块install报依赖模块的包找不到的问题 如果被依赖项⽬是springboot项⽬&#xff0c;那么可以把相关的springboot的东西移除掉&#xff0c;改造成普通项⽬。如果不想改造项⽬&#xff0c;那就添加部分的配置&#xff0c;因为springboot项⽬打包的时候会⽣…

uniapp如何获取安卓原生的Intent对象

通过第三方app唤起&#xff0c;并且获取第三方app唤起时携带的参数 因为应用a唤起应用b时&#xff0c;应用b第一时间就要拿到参数token&#xff0c;所以需要将获取参数的方法写在APP.vue中的onLaunch钩子里,如果其他地方要用可以选择vuex或者采用本地缓存。 uniapp中plus.run…

《多端统一的终极答案:X5内核增强版的渲染优化全解析》

跨端应用的需求呈爆发式增长&#xff0c;无论是电商购物、社交互动&#xff0c;还是金融理财类应用&#xff0c;都期望能够在不同平台上为用户提供一致且流畅的体验。而在这一过程中&#xff0c;跨端渲染技术成为了关键瓶颈。腾讯X5内核增强版的出现&#xff0c;犹如一道曙光&a…

深入理解算力:从普通电脑到宏观计算世界

在科技飞速发展的当下&#xff0c;“算力” 一词频繁出现在我们的视野中&#xff0c;无论是前沿的人工智能领域&#xff0c;还是新兴的区块链世界&#xff0c;算力都扮演着至关重要的角色。但对于大多数普通人来说&#xff0c;算力仿佛是一个既熟悉又陌生的概念。今天&#xff…

Paramiko复用 Transport 连接解析

1. 什么是 Transport 连接&#xff1f; 在 Paramiko 中&#xff0c;Transport 是负责底层 SSH 协议通信的核心类&#xff0c;它封装了以下功能&#xff1a; 加密通信&#xff1a;处理 SSH 协议的加密和解密。会话管理&#xff1a;维护与远程服务器的 TCP 连接。多路复用&…

sd webui 安装插件sd-webui-EasyPhoto依赖安装失败解决办法

在最新版的SD webui中&#xff0c;可以安装easyphoto插件&#xff0c;官方建议通过github安装&#xff0c;对无法科学上网的用户很不友好。对我自己来说是通过地址&#xff1a; https://gitee.com/wowai/sd-webui-EasyPhoto.git 分支&#xff1a;anyid 点击安装即可。 在安装…

WEBSTORM前端 —— 第2章:CSS —— 第3节:背景属性与显示模式

目录 1.Emmet写法 2.背景属性 &#xff08;1&#xff09; background-color &#xff08;2&#xff09; background-image &#xff08;3&#xff09; background-repeat &#xff08;4&#xff09;background-position &#xff08;5&#xff09;background-size &…