Java 面试宝典:volatile 的使用场景有哪些?

大家好,我是大明哥,一个专注「死磕 Java」系列创作的硬核程序员。
本文已收录到我的技术网站:https://skjava.com。有全网最优质的系列文章、Java 全栈技术文档以及大厂完整面经


回答

volatile 是一种轻量级的同步机制,它能保证共享变量的可见性,同时禁止重排序保证了操作的有序性,但是它无法保证原子性。所以使用 volatile 必须要满足这两个条件:

  1. 写入变量不依赖当前值。
  2. 变量不参与与其他变量的不变性条件。

volatile 比较适合多个线程读,一个线程写的场合,典型的场景有如下几个:

  1. 状态标志
  2. 重检查锁定的单例模式
  3. 开销较低的“读-写锁”策略

详解

volatile 使用条件

要想正确安全地使用 volatile ,必须要具备这两个条件:

  • 写入变量不依赖当前值:变量的新值不能依赖于之前的旧值。如果变量的当前值与新值之间存在依赖关系,那么仅使用 volatile 是不够的,因为它不能保证一系列操作的原子性。比如 i++。
  • 变量不参与与其他变量的不变性条件:如果一个变量是与其他变量共同参与不变性条件的一部分,那么简单地声明变量为 volatile 是不够的。

第一个条件很好理解,第二个条件这里需要解释下。

“变量不参与与其他变量的不变性条件”,这里的“不变性条件”指的是一个或多个变量在程序执行过程中需要保持的条件或关系,以确保程序的正确性。假设我们有两个变量,它们需要满足某种关系(例如,a + b = 99)。我们需要在多线程环境下保证这种关闭在任何时候都是成立的。如果这个时候我们只是将其中一个变量声明为 volatile,虽然确保了这个变量的更新对其他线程立即可见,但却不能保证这两个变量作为一个整体满足特定的不变性条件。在更新这两个变量的过程中,其他线程可能会看到这些变量处于不一致的状态。在这种情况下我们就需要使用锁或者其他同步机制来保证这种关系的整体一致性。

volatile 使用场景

volatile 比较适合多个线程读,一个线程写的场合

状态标志

当我们需要用一个变量来作为状态标志,控制线程的执行流程时,使用 volatile 可以确保当一个线程修改了这个标志时,其他线程能够立即看到最新的值。

public class TaskRunner implements Runnable {private volatile boolean running = true;  // 状态标志,控制任务是否继续执行public void run() {while (running) {  // 检查状态标志// 执行任务doSomething();}}public void stop() {running = false;  // 修改状态标志,使得线程能够停止执行}private void doSomething() {// 实际任务逻辑}
}

DCL 的单例模式

在实现单例模式时,为了保证线程安全,通常使用双重检查锁定(Double-Checked Locking)模式。在这种模式中,volatile 用于避免单例实例的初始化过程中的指令重排序,确保其他线程看到一个完全初始化的单例对象,具体来说,就是使用 volatile防止了Java 对象在实例化过程中的指令重排,确保在对象的构造函数执行完毕之前,不会将 instance 的内存分配操作指令重排到构造函数之外。

public class Singleton {// 使用 volatile 保证实例的可见性和有序性private static volatile Singleton instance;private Singleton() {}public static Singleton getInstance() {if (instance == null) {  // 第一次检查,避免不必要的同步synchronized (Singleton.class) {  // 锁定if (instance == null) {  // 第二次检查,确保只创建一次实例instance = new Singleton();}}}return instance;}
}

开销较低的“读-写锁”策略

这种策略一般都是允许多个线程同时读取一个资源,但只允许一个线程写入的同步机制。这种“读-写锁”非常适合读多写少的场景,我们可以利用 volatile + 锁的机制减少公共代码路径的开销。如下:

public class VolatileTest {  private volatile int value;  //读,不加锁,提供效率 public int getValue() {   return value;   }   //写操作,使用锁,保证线程安全  public synchronized int increment() {  return value++;  }  
}

在 J.U.C 中,有一个采用“读-写锁”方式的类:ReentrantReadWriteLock,它包含两个锁:一个是读锁,另一个是写锁。

下面是伪代码:

import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;public class DataStructure {private final ReadWriteLock readWriteLock = new ReentrantReadWriteLock();private final Object data = ...; // 被保护的数据public void read() {readWriteLock.readLock().lock(); // 获取读锁try {// 执行读操作// 例如,读取data的内容} finally {readWriteLock.readLock().unlock(); // 释放读锁}}public void write(Object newData) {readWriteLock.writeLock().lock(); // 获取写锁try {// 执行写操作// 例如,修改data的内容} finally {readWriteLock.writeLock().unlock(); // 释放写锁}}
}
  • 读操作 :多个线程可以同时持有读锁,因此多个线程可以同时执行 read() 方法。
  • 写操作: 只有一个线程可以持有写锁,并且在持有写锁时,其他线程不能读取或写入。

这种“读-写锁”策略提高了在多线程环境下对共享资源的读取效率,尤其是在读操作远远多于写操作的情况下。但是,它也会让我们的程序变更更加复杂,比如潜在的读写锁冲突、锁升级(从读锁升级到写锁)等问题。因此,在实际应用中,大明哥推荐直接使用 ReentrantReadWriteLock 即可,无需头铁自己造轮子。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/762188.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Batch Normalization(批量归一化)和 Layer Normalization(层归一化)

Batch Normalization(批量归一化)和 Layer Normalization(层归一化)都是深度学习中用于改善网络训练过程的归一化技术。尽管它们的目标相似,即通过规范化中间层的激活值来加速训练过程并提高性能,但它们在细节上有所不同。 Batch Normalization (批量归一化) Batch Nor…

谷歌地图TMS地图服务地址收集2024,测试可用

对于普通的开发者或者GIS从业者来说,免费的底图影像服务,太重要了。之前写过一篇谷歌地图的TMS地址收集的博文,由于谷歌网站关闭已经不能用。最近又发现了谷歌在国内开放了其他地址,在这里给大家分享一下。 https://gac-geo.googl…

Ant Design Vue和VUE3下的upload组件使用以及文件预览

Ant Design Vue和VUE3下的upload组件使用以及文件预览 文章目录 Ant Design Vue和VUE3下的upload组件使用以及文件预览一、多文件上传1.需求2.样例3.代码 二、单文件上传1. 需求2. 样例3.代码 二、多文件上传产生的时间超时问题三、文件系统名称更改1. 修改文件index.html2. 修…

【Java初阶(三)】方法的使用

❣博主主页: 33的博客❣ ▶文章专栏分类: Java从入门到精通◀ 🚚我的代码仓库: 33的代码仓库🚚 目录 1.前言2.方法的概念2.1方法定义2.2 实参和形参的关系 3. 方法的重载3.1方法重载的概念 4.递归4.1递归的概念4.2递归过程分析4.3 递归练习 5.总结 1.前言…

java核心面试题解析

1.索引 1.1创建索引: create index 索引名称 on 某张表 (列名) 示例: create index index_name on table (Column names) 1.2索引优化 MySQL数据库索引优化是提高查询性能的重要手段。以下是一些关键的索引优化策略: 选择正确的索引列: 经常需要排序、分组和联…

Leetcode热题100:图论

Leetcode 200. 岛屿数量 深度优先搜索法: 对于这道题来说,是一个非常经典的图的问题,我们可以先从宏观上面来看问题,也就是说在不想具体算法的前提下,简单的说出如何找到所有的岛屿呢? 如图中所示&#x…

win git filter-repo教程

git filter-repo 是一个用于过滤和清理 Git 仓库历史的工具,它可以高效地批量修改提交历史中的文件内容、删除文件、重命名文件以及进行其他历史重构操作。相较于 git filter-branch,它通常更快且更易于使用。 以下是一个基本示例,说明如何使…

oracle 19c单机版本补丁升级

文章目录 一、补丁包概述二、备份opatch三、替换高版本opatch四、打DB补丁1、关闭数据库2、关闭监听3、解压补丁4、冲突检测5、补丁空间检查6、执行补丁升级7、将更新内容加载到数据库8、最后查看数据库版本9、卸载补丁包 一、补丁包概述 补丁升级包 链接:https://…

【系统架构设计师】计算机系统基础知识 03

系统架构设计师 - 系列文章目录 01 系统工程与信息系统基础 02 软件架构设计 03 计算机系统基础知识 文章目录 系统架构设计师 - 系列文章目录 文章目录 前言 一、计算机系统概述 1.计算机组成 ​编辑2.存储系统 二、操作系统 ★★★★ 1.进程管理 2.存储管理 1.页式存储 …

Golang Gorm 自动分批查询

场景: 目标查询全量数据,但需要每次Limit分批查询,保护数据库 文档: https://gorm.io/zh_CN/docs/advanced_query.html // Param: // dest 目标地址 // batchSize 大小 // fc 处理函数func (db *DB) FindInBatc…

安卓 Android Activity 生命周期

文章目录 Intro生命周期方法 & 执行顺序结论code Intro 本文提供一个测试类通过打印的方式展示在多个Activity之间互相跳转的时候,各个Activity的生命周期相关方法的执行顺序。 生命周期方法 & 执行顺序结论 下图出自 郭霖 《第一行代码(第二…

速盾:免备案cdn的好处

免备案CDN(Content Delivery Network)是指不需要进行备案手续即可使用的CDN服务。备案是指在中国大陆地区提供互联网信息服务的网站必须向相关部门进行备案登记,以确保其合法合规的运营。 那么,免备案CDN有哪些好处呢&#xff1f…

电网的正序参数和等值电路(一)

本篇为本科课程《电力系统稳分析》的笔记。 本篇为第二章的第一篇笔记。 电力系统正常运行中,可以认为系统的三相结构和三相负荷完全对称。而对称三相的计算可以用一相来完成,其中所有给出的标称电压都是线电压的有效值,假定系统全部是Y-Y型…

深入了解23种设计模式:程序员必读指南

文章目录 引言概述基本原则设计模式总览 引言 随着编码时间拉长,遇到的问题增加,发现设计模式对于解决某类场景问题确实帮助很大。其实在不了解设计模式之前,其设计思想也已经在日常开发中有所体现,只是没有总结出来。设计模式像是…

C语言-常量

什么是常量? 答:常量是在程序执行过程中,其值不发生改变的量,常量分为直接常量和符号常量两种。 其中直接常量又可以分为整型常量、实型常量、字符型常量、字符串常量。 直接常量 1.整型常量 整型常量即整数,包括正整数,负整数和0。c语言中常量可以用八进制,十进制和十六…

【网站项目】291校园疫情防控系统

🙊作者简介:拥有多年开发工作经验,分享技术代码帮助学生学习,独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。🌹赠送计算机毕业设计600个选题excel文件,帮助大学选题。赠送开题报告模板&#xff…

The 2023 Guangdong Provincial Collegiate Programming Contest

I. Path Planning 嗯,怎么说呢,一般二维图,数据不是很大的比如n*m*log级别允许的,如果一眼不是bfs,可以考虑结合一下二分 本题可知,只能向下或者向右,那么我们就像如果答案为x,那么…

windows下使用压缩包安装mysql8.0数据库

获取安装包 可以访问mysql 官网下载压缩安装包 (官网地址:https://downloads.mysql.com/archives/community/) 根据自己的需要,下载对应mysql版本,我选择是是8.0.16版本 安装 解压之后,可以看到压缩包…

新概念英语1:Lesson7内容详解

新概念英语1:Lesson7内容详解 如何询问人的个人信息 本课里有两个关于个人信息的问句,一个是问国籍,一个是问工作,句型如下: what nationality are you?询问国籍 回复一般就是我是哪国人,I’m Chinese…

文章解读与仿真程序复现思路——中国电机工程学报EI\CSCD\北大核心《基于老化成本实时次梯度的异构储能系统功率分配策略》

本专栏栏目提供文章与程序复现思路,具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…