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 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 对象自身为锁

CA.LA
CA.LA

第一款时尚产品在线设计平台,服装设计系统

下载
public class Account {
    private static final List 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 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 log = Collections.synchronizedList(new ArrayList<>());
// 或更现代的选择(JDK 14+):
// private static final List 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 处理金钱:这是金融系统硬性规范。

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

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

832

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

738

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

734

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

2

2026.01.16

热门下载

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

精品课程

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

共23课时 | 2.5万人学习

C# 教程
C# 教程

共94课时 | 6.8万人学习

Java 教程
Java 教程

共578课时 | 46.3万人学习

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

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