当前位置: 首页 > news >正文

【Java】 volatile 和 synchronized 的比较及使用场景

在 Java 的并发编程中,volatile 和 synchronized 是两个常用的关键字,它们分别用于保证多线程环境中的 可见性原子性,但它们的工作原理和适用场景却有所不同。今天,我们将深入探讨这两个关键字的异同,帮助大家理解它们的使用场景和选择方式。

1. volatile 关键字

功能简介

volatile 是 Java 中的一个轻量级同步机制,它用来确保 变量的可见性。当一个线程修改了 volatile 变量的值,其他线程能够立即看到修改后的值。它通过 内存屏障 来禁止编译器和 CPU 对 volatile 变量的重排序,确保每次读取都是从主内存中获取最新的值。

使用场景

volatile 主要用于以下两种场景:

  • 状态标志:通常用于线程之间的状态通信。例如,线程池中常用一个 volatile 标志来通知线程是否需要停止执行。
  • 单一变量的可见性:当你需要保证多个线程对同一变量的修改是可见的,而不需要操作的原子性时,volatile 是一个不错的选择。

示例代码

public class VolatileExample {private volatile boolean flag = false;public void setFlag() {flag = true;}public boolean checkFlag() {return flag;}
}

在这个例子中,volatile 确保了 flag 变量在多个线程中是可见的,即当一个线程修改了 flag 的值,其他线程能够立即看到这个变化。

优缺点

优点

  • 轻量级,性能较好。
  • 简单易用,避免了使用 synchronized 带来的开销。

缺点

  • 只保证 可见性,不能保证 原子性。对于复合操作(如 i++),volatile 并不适用。

2. synchronized 关键字

功能简介

synchronized 是 Java 中的同步机制,它可以保证 原子性可见性。使用 synchronized 可以确保在同一时刻只有一个线程可以执行某个方法或代码块,其他线程必须等待锁的释放。synchronized 确保了对共享资源的访问是安全的。

使用场景

synchronized 适用于需要对共享资源进行 复合操作 的场景,尤其是当多个线程同时修改共享数据时。例如:

  • 多线程环境下对共享数据的读写操作
  • 确保操作的原子性,避免数据的竞争和不一致。

示例代码

public class SynchronizedExample {private int count = 0;public synchronized void increment() {count++;}public synchronized int getCount() {return count;}
}

在这个例子中,synchronized 确保了 incrementgetCount 方法在多线程环境下的线程安全。只有一个线程能够访问这两个方法,避免了并发问题。

优缺点

优点

  • 保证 原子性可见性,适用于复杂的并发操作。
  • 适用于保护临界区代码,防止数据竞态。

缺点

  • 性能开销较大,每次进入 synchronized 方法或代码块时,都需要获取锁,离开时需要释放锁。这会导致线程上下文切换,影响性能。
  • 容易引发 死锁 问题,如果不小心使用锁的顺序,会导致线程互相等待,造成程序无法继续执行。

3. volatilesynchronized 的对比

特性volatilesynchronized
保证的功能仅保证变量的 可见性保证 原子性可见性
适用场景适用于简单的标志变量(如停止线程标志)适用于需要对共享资源进行原子操作的场景
性能开销性能较好,无锁开销性能开销较大,需要线程上下文切换和加锁操作
原子性不保证(例如自增操作 i++ 是非原子操作)保证(确保对共享变量的原子修改)
实现方式通过内存屏障确保可见性通过锁机制保证互斥访问
使用简单性简单,只涉及变量声明和使用需要额外的加锁和解锁操作
适用于复合操作不适合复合操作(例如自增、判断并更新等)适合复合操作(如自增、自减等复杂操作)
线程竞争无锁开销,但只能保证可见性,无法避免线程竞争可以避免线程竞争,保证操作的完整性

4. 何时选择 volatile,何时选择 synchronized

选择 volatile

  • 简单的变量控制:如果你只需要确保变量的可见性,并且操作是简单的读写操作(如状态标志),volatile 是更优的选择。
  • 性能敏感的场景:由于 volatile 没有锁的开销,性能上会优于 synchronized

选择 synchronized

  • 复合操作:如果你的操作涉及多个步骤(例如 i++,或多个值的修改),volatile 无法保证原子性,这时应该使用 synchronized
  • 多线程共享数据的场景:在多个线程同时修改同一数据时,synchronized 可以确保数据的安全性和一致性。

5. 总结

volatilesynchronized 是 Java 中两种常用的并发控制机制。volatile 提供了轻量级的可见性保障,适用于简单的场景,但不能保证原子性;而 synchronized 提供了强大的原子性保障,适用于复杂的并发操作,但会带来较大的性能开销。

在实际开发中,选择哪种机制取决于你的具体需求:

  • 如果只是需要确保变量的可见性,使用 volatile 更为高效。
  • 如果需要保证多个操作的原子性或保护共享资源的访问,使用 synchronized 更为合适。

理解它们的原理和适用场景,将帮助你在并发编程中做出更加合理的选择。

相关文章:

  • matlab插值方法(简短)
  • 可编辑38页PPT | 数字化建设—财务系统规划方案
  • Git仓库迁移
  • 电流检测放大器的优质选择XBLW-INA180/INA181
  • MQTT详细介绍
  • SQL注入 ---04
  • 移动端巡检点检,让设备管理更便捷高效
  • centos服务器,疑似感染phishing家族钓鱼软件的检查
  • ElasticSearch高级功能
  • OpenUCX 库介绍与使用指南
  • 【信息系统项目管理师】第5章:信息系统工程 - 36个经典题目及详解
  • 生成对抗网络(Generative Adversarial Networks ,GAN)
  • Python笔记:在环境变量中增加了dll加载路径,python提示DLL加载失败
  • 匿名函数lambda、STL与正则表达式
  • 最小二乘拟合曲线
  • bat——自动重启程序
  • 酒店行业冰与火:一边流拍,一边扩张
  • 第22篇:Linux系统的Switch字符设备驱动设计
  • 数据的模型分析及可视化
  • 【python机器学习】Day 25 异常处理
  • 占地57亩的“潮汕豪宅”面临强制拆除:曾被实施没收,8年间举行5次听证会
  • 外企聊营商|特雷通集团:税务服务“及时雨”
  • 学习教育期间违规吃喝,李献林、叶金广等人被通报
  • 特朗普访中东绕行以色列,专家:凸显美以利益分歧扩大
  • 铁路部门:确保沿线群众安全,焦柳铁路6个区段将陆续安装防护栅栏
  • 庆祝上海总工会成立100周年暨市模范集体劳动模范和先进工作者表彰大会举行,陈吉宁寄予这些期待