0

0

在Java中如何实现线程安全的数据共享_Java多线程数据一致性实现解析

P粉602998670

P粉602998670

发布时间:2026-01-23 12:22:20

|

650人浏览过

|

来源于php中文网

原创

synchronized块保护共享变量最直接,需锁最小临界区并用私有锁对象;volatile仅保可见性不保原子性;atomicinteger等并发工具更高效;threadlocal实现线程隔离而非共享。

在java中如何实现线程安全的数据共享_java多线程数据一致性实现解析

synchronized 块保护共享变量最直接

当多个线程读写同一个对象字段(如 counter++)时,非原子操作会导致数据丢失。加 synchronized 是最基础且可控的方案,关键不是锁整个方法,而是锁最小临界区。

  • 优先用私有锁对象(private final Object lock = new Object();),避免锁 this 或类对象引发意外竞争
  • 不要在同步块里调用外部可变对象的方法(比如 list.sort(...)),可能造成锁升级或死锁
  • synchronized 无法解决“先检查后执行”(check-then-act)问题,例如 if (list.isEmpty()) list.add(x); 需整体包裹
private int counter = 0;
private final Object lock = new Object();

public void increment() {
    synchronized (lock) {
        counter++; // 这才是原子的
    }
}

java.util.concurrent 包里的工具类更高效

对计数、队列、Map 等常见结构,JDK 提供了无锁或分段锁实现,性能远高于粗粒度 synchronized,且语义清晰。

  • AtomicInteger 适合简单状态计数,但注意 getAndIncrement()incrementAndGet() 返回值不同
  • ConcurrentHashMap 不允许 null 作为 key 或 value,否则抛 NullPointerException
  • CopyOnWriteArrayList 适合读多写少场景,但每次写都复制数组,写频繁时内存和 GC 压力大
private AtomicInteger counter = new AtomicInteger(0);
private ConcurrentHashMap<String, String> cache = new ConcurrentHashMap<>();

public void addCache(String key, String value) {
    cache.putIfAbsent(key, value); // 原子性保证,无需额外同步
}

ThreadLocal 不是共享,而是“伪共享”隔离

很多人误以为 ThreadLocal 能解决共享数据一致性问题,其实它恰恰相反:每个线程持有一份独立副本。适用于“线程内单次使用、避免参数层层传递”的场景,比如数据库连接、用户上下文。

PHP5 和 MySQL 圣经
PHP5 和 MySQL 圣经

本书是全面讲述PHP与MySQL的经典之作,书中不但全面介绍了两种技术的核心特性,还讲解了如何高效地结合这两种技术构建健壮的数据驱动的应用程序。本书涵盖了两种技术新版本中出现的最新特性,书中大量实际的示例和深入的分析均来自于作者在这方面多年的专业经验,可用于解决开发者在实际中所面临的各种挑战。

下载
  • ThreadLocal 变量不清理会导致内存泄漏(尤其在线程池中),务必在 finally 块或使用 try-with-resources 风格调用 remove()
  • 不能用于跨线程传递数据;子线程不会自动继承父线程的 ThreadLocal 值,需显式用 InheritableThreadLocal
  • 它的 set()get() 操作本身是线程安全的,但内部对象(如 SimpleDateFormat)仍需自行保证线程安全
private static final ThreadLocal<SimpleDateFormat> formatter =
    ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd"));

public String formatDate(Date date) {
    try {
        return formatter.get().format(date);
    } finally {
        // formatter.remove(); // 忘记这句,在线程复用时可能污染后续请求
    }
}

volatile 仅保证可见性,不保证原子性

volatile 关键字常被误当作“轻量级同步”,但它只做两件事:禁止指令重排序 + 强制每次读写都走主内存。对复合操作(如 i++)完全无效。

立即学习Java免费学习笔记(深入)”;

  • 适用场景很窄:布尔开关标志(running = false)、一次性发布(如单例双重检查中的 instance 字段)
  • 不能替代 synchronized 或原子类,volatile int counter 依然会因竞态导致结果错误
  • 在 Java 5+ 中语义已明确,但低版本 JVM 行为不可靠,慎用于老环境
private volatile boolean shutdownRequested = false;

public void shutdown() {
    shutdownRequested = true; // ✅ 安全:单纯写入
}

public void doWork() {
    while (!shutdownRequested) { // ✅ 安全:单纯读取
        // ...
    }
}
真正难的是识别哪些状态必须强一致、哪些可以最终一致。比如缓存更新失败是否允许短暂脏读,日志计数是否允许丢失几条——这些业务语义决定了该用 ReentrantLock 还是 StampedLock,甚至要不要放弃强一致性改用消息队列解耦。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1089

2024.03.01

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

846

2023.08.22

sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

409

2023.09.04

sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

409

2023.09.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

608

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

334

2025.08.29

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

22

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号