java volatile变量及其使用场景

2024-09-03 00:32

本文主要是介绍java volatile变量及其使用场景,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

java中的一种稍弱的同步机制,就是volatile变量,用于确保将变量的更新操作通知到其他线程。

变量声明为volatile后:

(1)编译器与运行时都会注意到这个变量是共享的,因此不会将该变量上的操作与其他内存操作一起重排序(重排序不懂的,可以自行百度,需要理解)。

(2)volatile变量不会被缓存在寄存器或对其他处理器不可见的地方

因此volatile变量总是会返回最新的值。

volatiel是比synchronize关键字更轻量级的同步机制,因为它不需要加锁,也就不会阻塞,所以性能会好点。

这边为了加深对volatile变量总能返回最新值的理解,你可以这样想:从内存可见性角度来看,写入volatile相当于推出同步代码块,而读取volatile变量就相当于进入同步代码块。

注意:并不建议过度依赖volatile变量提供的可见性。如果在代码中依赖volatile变量来控制状态的可见性,通常比使用锁的代码更脆弱,更难理解。

下面我来介绍一下volatile的使用场景,当且仅当满足以下所有条件时,才应该使用volatile变量:

(1) 对变量的写入操作不依赖变量的当前值,或者你能确保只有单个线程更新变量的值。

(2) 该变量不会与其他状态变量一起纳入不变性条件。

(3) 在访问变量时不需要加锁

可能你有点不好理解上面三句话的意思,我们一条一条来讲

对于第(1)条的理解就是:一般我们可能有这样的操作习惯,就是检查某个状态以判断是否退出循环,看一下下面这个示例,线程试图通过类似于数绵羊的传统方法进入休眠状态。为了使这个示例能正确执行,asleep必须为volatile变量。否则,当asleep被另一个线程修改时,执行判断的线程却发现不了。我们也可以用锁来确保asleep更新操作的可见性,但这将会使得代码变得更复杂。什么叫 对变量的写入操作不依赖变量的当前值?就如示例1中decide该函数中对asleep先判断后执行(执行中保包含了对变量的改变),这种情况下没使用锁,所以可能在多线程情况下,alseep的内存可见性得不到保证。对于这类“对变量的写入操作依赖变量的当前值”的操作,是需要用锁进行保护的,此时可以考虑用内置锁或者显示锁,但是方便起见可以使用volatile来保护asleep这个变量。毕竟只是想保护一个变量。解决方式如示例2,当然该示例对上述第三个条件同样满足。那条件2中“该变量不会与其他状态一起纳入不变性条件”是什么意思呢?就是volatile变量不能与其他共享变量做复合操作,volatile变量本身在没有锁保护的情况下也不能做复合操作(这是我的理解)。

 

示例1:

public class NoVolatile {private int sheepCount;private boolean asleep;public void decide() {while (! asleep) {countSomeSheep();}}public void setAsleep() {asleep = true;}private void countSomeSheep() {if (sheepCount < 1000) {sheepCount++;} else {sheepCount = 0;setAsleep();}}
}

示例2:

class VolatileDemo {private volatile asleep;public void decide() {while (! asleep) {countSomeSheep();}}public void setAsleep() {asleep = true;}
}

对于volatile变量的使用需要有以下注意点:

加锁机制既可以确保可见性又可以确保原子性,而volatile变量只能确保可见性。啥意思呢,就是,volatile的语义不足以确保递增操作(count++)的原子性,除非你能确保只有一个线程对变量执行写操作。

 

 

这篇关于java volatile变量及其使用场景的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

LiteFlow轻量级工作流引擎使用示例详解

《LiteFlow轻量级工作流引擎使用示例详解》:本文主要介绍LiteFlow是一个灵活、简洁且轻量的工作流引擎,适合用于中小型项目和微服务架构中的流程编排,本文给大家介绍LiteFlow轻量级工... 目录1. LiteFlow 主要特点2. 工作流定义方式3. LiteFlow 流程示例4. LiteF

使用Python开发一个现代化屏幕取色器

《使用Python开发一个现代化屏幕取色器》在UI设计、网页开发等场景中,颜色拾取是高频需求,:本文主要介绍如何使用Python开发一个现代化屏幕取色器,有需要的小伙伴可以参考一下... 目录一、项目概述二、核心功能解析2.1 实时颜色追踪2.2 智能颜色显示三、效果展示四、实现步骤详解4.1 环境配置4.

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

java中long的一些常见用法

《java中long的一些常见用法》在Java中,long是一种基本数据类型,用于表示长整型数值,接下来通过本文给大家介绍java中long的一些常见用法,感兴趣的朋友一起看看吧... 在Java中,long是一种基本数据类型,用于表示长整型数值。它的取值范围比int更大,从-922337203685477

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔

SQL中JOIN操作的条件使用总结与实践

《SQL中JOIN操作的条件使用总结与实践》在SQL查询中,JOIN操作是多表关联的核心工具,本文将从原理,场景和最佳实践三个方面总结JOIN条件的使用规则,希望可以帮助开发者精准控制查询逻辑... 目录一、ON与WHERE的本质区别二、场景化条件使用规则三、最佳实践建议1.优先使用ON条件2.WHERE用

java Long 与long之间的转换流程

《javaLong与long之间的转换流程》Long类提供了一些方法,用于在long和其他数据类型(如String)之间进行转换,本文将详细介绍如何在Java中实现Long和long之间的转换,感... 目录概述流程步骤1:将long转换为Long对象步骤2:将Longhttp://www.cppcns.c

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte

SpringBoot服务获取Pod当前IP的两种方案

《SpringBoot服务获取Pod当前IP的两种方案》在Kubernetes集群中,SpringBoot服务获取Pod当前IP的方案主要有两种,通过环境变量注入或通过Java代码动态获取网络接口IP... 目录方案一:通过 Kubernetes Downward API 注入环境变量原理步骤方案二:通过

Springboot整合Redis主从实践

《Springboot整合Redis主从实践》:本文主要介绍Springboot整合Redis主从的实例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言原配置现配置测试LettuceConnectionFactory.setShareNativeConnect