Java线程安全是什么?安全访问详解
时间:2026-05-14 12:10:39 184浏览 收藏
Java线程安全的本质并非简单地“加锁”,而是确保多线程并发读写共享数据时结果始终符合预期——哪怕99%的时间看似正常,只要一次出现count++少加、时间解析错乱或内存泄漏等异常,就已暴露不安全;文章深入剖析了count++非原子性的底层原因(load-add-save三步可被中断),指出volatile无法解决竞态,synchronized适合快速修复但需注意锁粒度与风险,而高并发场景下ReentrantLock和AtomicInteger各具优势;更关键的是提醒开发者:真正的隐患往往藏在那些“看似无共享”的地方——如SimpleDateFormat、静态StringBuilder缓存、ThreadLocal未清理等,稍不留神就会引发难以复现的并发Bug。

线程安全不是“有没有锁”,而是“多线程同时读写共享数据时,结果是否始终符合预期”。只要一次运行出错(比如 count++ 本该加 10000 次却只加了 8327),就是不安全——哪怕它大多数时候看起来“没问题”。
为什么 count++ 不是原子操作?
它在 CPU 层面实际拆成三步:load(从内存读值到寄存器)、add(寄存器+1)、save(写回内存)。两个线程可能同时 load 到同一个旧值,各自 +1 后都 save,最终只加了一次。
- 现象:两个线程各执行 5000 次
count++,打印结果常为7xxx或9xxx,而非10000 - 根本原因:指令执行顺序被线程调度打乱,且 JVM 不保证这三步不可中断
- 注意:
volatile能让save立即对其他线程可见,但解决不了load→add→save中间被插队的问题
用 synchronized 最快止血
适合快速修复、逻辑简单、并发压力不大的场景。本质是给临界区加一把 JVM 内置的“互斥锁”,同一时刻只放行一个线程。
public class Counter {
private int count = 0;
// 方式1:同步方法(锁的是 this 对象)
public synchronized void increment() {
count++;
}
// 方式2:同步代码块(推荐,锁粒度更可控)
private final Object lock = new Object();
public void incrementFine() {
synchronized (lock) {
count++;
}
}
}
- 别用
public synchronized static void锁类对象,除非真要全局串行 - 避免在同步块里调用外部方法(如
System.out.println()),防止锁被意外持有过久 - 同步方法和同步代码块性能差异不大,但后者能明确锁对象,便于排查死锁
高并发或需要控制力时选 ReentrantLock 或 AtomicInteger
ReentrantLock 提供超时、可中断、公平性等能力;AtomicInteger 基于 CPU 的 CAS 指令,无锁、轻量,但仅适用于单变量简单操作。
// ReentrantLock 示例
private final ReentrantLock lock = new ReentrantLock();
public void incrementWithLock() {
lock.lock();
try {
count++;
} finally {
lock.unlock(); // 必须在 finally 中释放
}
}
// AtomicInteger 示例
private final AtomicInteger atomicCount = new AtomicInteger(0);
public void incrementAtomic() {
atomicCount.incrementAndGet(); // 原子性保障,无需锁
}
ReentrantLock忘记unlock()会导致永久阻塞——务必套try/finallyAtomicInteger的incrementAndGet()是原子的,但if (atomicCount.get() 这种“读-改-写”仍需额外同步- 不要为了“高级感”强行用
ReentrantLock替换synchronized,JVM 对后者做了大量优化
真正容易被忽略的,是那些“看起来没共享”的状态:比如用 SimpleDateFormat 解析时间、静态工具类里缓存了 StringBuilder、甚至 ThreadLocal 忘记 remove() 导致内存泄漏——它们都在悄悄制造线程安全问题。
好了,本文到此结束,带大家了解了《Java线程安全是什么?安全访问详解》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!
相关阅读
更多>
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
200 收藏
-
128 收藏
-
148 收藏
-
174 收藏
-
184 收藏
-
467 收藏
-
228 收藏
-
434 收藏
-
279 收藏
-
185 收藏
-
296 收藏
-
259 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习