0

0

Java 中线程同步的正确实践:为何仅同步方法不足以保护静态共享资源

心靈之曲

心靈之曲

发布时间:2026-01-03 15:17:02

|

773人浏览过

|

来源于php中文网

原创

Java 中线程同步的正确实践:为何仅同步方法不足以保护静态共享资源

本文详解 java 多线程环境下对静态共享集合(如 static list)的同步策略,指出仅使用 synchronized 方法无法保证线程安全,并通过对比分析、代码示例与关键注意事项,阐明必须显式锁定共享对象本身的原因。

在 Java 并发编程中,synchronized 是最基础且常用的同步机制,但其作用范围和锁对象的选择极易被误解。以 Account 类为例,初看之下,将 deposit() 和 withdraw() 声明为 synchronized 方法似乎已确保了线程安全——毕竟所有对 balance 和 log 的修改都被“串行化”了。然而,这种理解只在锁对象与被保护资源严格对应时才成立;而本例中,锁对象(this 实例)与被保护资源(static List<Double> log)根本不同步

问题根源:锁对象错配

public synchronized void deposit(...) 等价于:

public void deposit(double val) {
    synchronized (this) {  // ← 锁的是当前 Account 实例!
        balance = balance + val;
        log.add(val);  // ← 但 log 是 static,被所有实例共享!
    }
}

这意味着:

  • 若有多个 Account 实例(如 a1、a2),线程 T1 调用 a1.deposit() 会锁住 a1;
  • 线程 T2 同时调用 a2.withdraw() 会锁住 a2;
  • 二者互不阻塞,却同时操作同一个静态 log 列表 → ArrayList 非线程安全,add() 操作可能引发 ConcurrentModificationException、数据丢失或内部结构损坏(如 size 字段未正确更新)。

这正是原方案不安全的根本原因:方法级同步保护的是实例状态(balance),而非跨实例共享的静态资源(log)

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

正确解法:按资源粒度精准加锁

要保护静态 log,必须让所有访问它的代码都竞争同一把锁。最佳实践是直接以 log 对象自身为锁

PixVerse
PixVerse

PixVerse是一款强大的AI视频生成工具,可以轻松地将多种输入转化为令人惊叹的视频。

下载
public class Account {
    private static final List<Double> log = new ArrayList<>(); // ✅ final 保证引用不可变
    private double balance;

    public Account() { balance = 0.0; }

    public synchronized void deposit(double val) {
        balance += val;
        synchronized (log) {  // ✅ 所有线程均竞争 log 这一共享锁
            log.add(val);
        }
    }

    public synchronized void withdraw(double val) {
        balance -= val;
        synchronized (log) {
            log.add(-val); // 注意:原题 withdraw 日志应记录负值,体现资金流出
        }
    }

    // 安全的静态访问器(可选)
    public static List<Double> getLog() {
        synchronized (log) {
            return new ArrayList<>(log); // ✅ 返回副本,避免外部直接修改
        }
    }
}
? 关键点说明:log 声明为 final:防止意外重新赋值导致锁失效(如 log = new ArrayList<>() 后,新旧引用锁对象不一致);双重锁嵌套可行:synchronized(this) 保护 balance,synchronized(log) 保护 log,互不干扰;避免锁 Class 对象(如 synchronized(Account.class)):虽能保护静态资源,但粒度过大,易造成不必要的线程阻塞。

更优替代方案:使用线程安全集合

对于日志等追加场景,推荐直接选用 java.util.concurrent 包中的线程安全集合,语义更清晰、性能通常更优:

private static final List<Double> log = Collections.synchronizedList(new ArrayList<>());
// 或更现代的选择(JDK 14+):
// private static final List<Double> log = new CopyOnWriteArrayList<>();

此时 log.add() 本身已线程安全,无需额外 synchronized(log) 块(但注意:Collections.synchronizedList 的迭代仍需手动同步)。

重要补充:业务逻辑陷阱

文中代码还存在一个严重设计缺陷:使用 double 表示货金额。浮点数精度误差会导致不可接受的财务偏差(例如 0.1 + 0.2 != 0.3)。生产环境必须改用 BigDecimal 或整数(以分为单位):

private BigDecimal balance = BigDecimal.ZERO;
public void deposit(BigDecimal val) {
    balance = balance.add(val);
    synchronized (log) {
        log.add(val.doubleValue()); // 若日志仅作调试,可保留 double;否则也应用 BigDecimal
    }
}

总结

  • 同步目标决定锁对象:保护实例变量 → 锁 this;保护静态变量 → 锁该静态变量(或 Class 对象);
  • 静态集合必须显式同步:synchronized 方法无法覆盖 static 资源;
  • 优先选用并发集合:CopyOnWriteArrayList、ConcurrentLinkedQueue 等比手写同步更可靠;
  • ⚠️ 永远避免 double/float 处理金钱:这是金融系统硬性规范。

正确的同步不是“加锁越多越好”,而是“锁得恰到好处”——精准匹配临界资源与锁粒度,方为高并发程序的稳健基石。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

595

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

108

2025.10.23

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

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

335

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

108

2025.10.23

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

186

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

134

2025.11.27

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

891

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

32

2025.12.06

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 82.1万人学习

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

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