0

0

在Java里多线程下如何保证数据一致性_Java同步策略设计说明

P粉602998670

P粉602998670

发布时间:2026-02-09 06:27:21

|

503人浏览过

|

来源于php中文网

原创

synchronized、ReentrantLock、AtomicInteger 和 volatile 各有适用场景:synchronized 保证临界区串行执行;ReentrantLock 支持可中断和超时;AtomicInteger 适用于单变量原子操作;volatile 仅保证可见性,不保证原子性。

在java里多线程下如何保证数据一致性_java同步策略设计说明

synchronized 锁住临界区是最直接的方式

当多个线程可能同时读写同一个共享变量(比如一个 int counter),不加控制就会出现“写丢失”——两个线程都读到 5,各自加 1 后都写回 6,结果本该是 7。用 synchronized 方法或代码块能强制串行执行:

常见写法有两种:
- 实例方法上加 synchronized:锁的是当前对象(this
- 静态方法上加 synchronized:锁的是类的 Class 对象
- 更灵活的是用 synchronized(obj) { ... },显式指定锁对象,避免锁太粗影响并发度

注意点:
- 锁对象不能是基本类型包装类(如 Integer)、字符串字面量或可变对象(如 ArrayList),否则容易因对象替换或字符串常量池导致锁失效
- 不要在同步块里调用外部可变对象的未知方法,防止死锁或阻塞延长

ReentrantLocksynchronized 多了可中断和超时能力

当需要“尝试获取锁但不想无限等待”,或者要响应线程中断时,ReentrantLock 是更合适的选择。它支持 tryLock()tryLock(long, TimeUnit)lockInterruptibly()

使用要点:
- 必须在 finally 块中调用 unlock(),否则可能永久持有锁
- 不要混用 synchronizedReentrantLock 保护同一资源,它们互不感知
- 如果只是简单计数,优先考虑 AtomicInteger 而不是锁,性能更高且无锁开销

AtomicInteger 替代锁做简单状态更新最轻量

对单个整型变量的自增、自减、比较并设置(CAS)等操作,AtomicInteger 是零锁方案。它底层依赖 CPU 的 CAS 指令,比锁快得多,也不存在死锁风险。

典型场景:
- 计数器(如请求统计、连接数)
- 状态标志(如 AtomicBoolean running = new AtomicBoolean(true)
- 序列号生成(配合 incrementAndGet()

限制:
- 只适用于单变量原子操作;多个变量联动更新(如“扣库存同时改订单状态”)不能靠多个原子类拼凑,仍需锁或事务
- getAndIncrement()incrementAndGet() 返回值不同,别用反了

volatile 关键字只保可见性,不保原子性

volatile 能让线程立刻看到其他线程对该变量的最新写入,但它**不能防止指令重排序带来的逻辑错误,也不能保证复合操作的原子性**。比如 flag = true 可以用 volatile,但 counter++ 不行——因为这是“读-改-写”三步,volatile 不拦中间那两步。

There’s An AI For That
There’s An AI For That

全球领先的 AI 聚合器,收集10,225个AI工具,可用于超过2,548个任务。

下载

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

适用场合:
- 纯状态标记(如 volatile boolean shutdownRequested
- 作为双重检查锁(DCL)中单例实例的修饰符(配合 synchronized
- 与 AtomicReference 配合实现无锁数据结构

常见误用:
- 把 volatile List 当线程安全集合用(add() 仍会出并发问题)
- 以为加了 volatile 就不用 synchronized 了,结果发现数值还是错的

实际项目里,多数一致性问题不是“选哪个机制”,而是“哪段逻辑真正需要保护”。先厘清共享数据的读写边界,再决定用 volatile、原子类、锁,还是干脆重构为无共享设计。锁粒度太粗吞吐低,太细又容易漏;原子类看着简洁,但组合操作一多就容易掉进非原子陷阱。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

355

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

35

2025.11.30

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1518

2023.10.24

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1518

2023.10.24

字符串常量的表示方法
字符串常量的表示方法

字符串常量的表示方法:1、使用引号;2、转义字符;3、多行字符串;4、原始字符串;5、字符串连接;6、字符串字面量和对象;7、编码问题。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

140

2023.12.26

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

444

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

213

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1518

2023.10.24

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

127

2026.02.06

热门下载

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

精品课程

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

共23课时 | 3.4万人学习

C# 教程
C# 教程

共94课时 | 9万人学习

Java 教程
Java 教程

共578课时 | 61.7万人学习

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

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