volatile关键字三重功效

2024-06-18 08:48

本文主要是介绍volatile关键字三重功效,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  1. 64位写入原子性

举一个简单的例子,对于一个long型变量的赋值和取值操作而言,在多线程场景下,线程A调用set(100),线程B调用get(),在某些场景下,返回值可能不是100。

package com.lc.test02;import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;/*** @author liuchao* @date 2020/7/3*/
public class Test01 {private long a = 0;public void set(long a) {//线程1设置值this.a = a;}public long get() {//线程2获取值,返回值可能不是100return a;}public static void main(String[] args) {Test01 test = new Test01();ExecutorService executor = Executors.newFixedThreadPool(2);CountDownLatch latch = new CountDownLatch(1);executor.execute(() -> {try {latch.await();} catch (InterruptedException e) {e.printStackTrace();}long i = 100L;test.set(i);System.out.println("-----设置:" + i);});executor.execute(() -> {try {latch.await();} catch (InterruptedException e) {e.printStackTrace();}System.out.println("----获取:" + test.get());});latch.countDown();executor.shutdown();}}

 为什么获取到的值有可能不是100呢,是因为JVM规范并未要求64位的long和double写入是原子性的,64位的数据有可能被拆分成两个32位的数据写入,这样另一个线程拿到的数据有可能是32位的不完整的数据,如果在long属性前面加上volatile关键字就可以解决此问题。

  1. 内存可见性

JVM将内存组织为主内存和工作内存两个部分。

针对主内存中的变量,线程A操作后线程B看到要经过的流程

线程A操作后数据存储在线程A工作内存=》save到主内存=》线程B从主内存load到线程B工作内存

那在整个操作过程中是非原子性操作的,有可能线程A修改后是10,但是线程B读取到的数据是非10,因为线程A修改后的数据还未save到主内存,那要解决这个问题也比较简单就是直接在属性前面加上volatile关键字,也就是解决了内存可见性问题 

  1. 禁止重排序

在经典的单线程安全的写法上DCL

//注意,此代码有安全问题package com.lc.test02;/*** @author liuchao* @date 2020/7/3*/
public class Test01 {private static Test01 instance;public static Test01 getInstance() {if (null == instance) {synchronized (instance) {//为了性能验证 使用lockif (null == instance) {instance = new Test01();//有问题的代码}}}return instance;}}

上述的instance = new Test01();代码有问题:其底层会分为三个操作:

(1)分配一块内存。

(2)在内存上初始化成员变量。

(3)把instance引用指向内存。 

在这三个操作中,操作(2)和操作(3)可能重排序,即先把instance指向内存,再初始化成员变量,因为二者并没有先后的依赖关系。此时,另外一个线程可能拿到一个未完全初始化的对象。这时,直接访问里面的成员变量,就可能出错。这就是典型的“构造函数溢出”问题。解决办法也很简单,就是为instance变量加上volatile修饰

 

这篇关于volatile关键字三重功效的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1071795

相关文章

Oracle Start With关键字

Oracle Start With关键字 前言 旨在记录一些Oracle使用中遇到的各种各样的问题. 同时希望能帮到和我遇到同样问题的人. Start With (树查询) 问题描述: 在数据库中, 有一种比较常见得 设计模式, 层级结构 设计模式, 具体到 Oracle table中, 字段特点如下: ID, DSC, PID; 三个字段, 分别表示 当前标识的 ID(主键), DSC 当

关键字synchronized、volatile的比较

关键字volatile是线程同步的轻量级实现,所以volatile性能肯定比synchronized要好,并且volatile只能修饰于变量,而synchronized可以修饰方法,以及代码块。随着JDK新版本的发布,synchronized关键字的执行效率上得到很大提升,在开发中使用synchronized关键字的比率还是比较大的。多线程访问volatile不会发生阻塞,而synchronize

JavaScript 根据关键字匹配数组项

要在JavaScript数组中根据关键字匹配项,可以使用filter方法结合一个测试函数。以下是一个示例代码,定义了一个函数findByKeyword,该函数接受一个数组和一个关键字,然后返回一个新数组,其中包含与关键字匹配的所有项。 function findByKeyword(array, keyword) {return array.filter(item => {// 假设要匹配的是对象

MySQL 的关键字

MySQL 中的关键字是数据库中具有特殊含义的保留字,它们用于定义数据库结构、操作数据库数据和控制数据库行为。关键字在 MySQL 查询中扮演着至关重要的角色,因为它们是 SQL 语句的核心组成部分。 1. 数据定义语言 (DDL) 关键字 数据定义语言 (DDL) 关键字用于定义、修改和删除数据库结构,如数据库、表和索引等。这些关键字通常用于创建、删除表结构以及修改表的列等操作。 1.1

C++中的mutable关键字详解

目录 1.概述 2.使用场景 3.示例 4.mutable修饰Lambda表达式 5.注意事项 1.概述         在C++中,mutable也是为了突破const的限制而设置的。被mutable修饰的变量,将永远处于可变的状态,即使在一个const函数中。         我们知道,被const关键字修饰的函数的一个重要作用就是为了能够保护类中的成员变量。即:该函数可以

[Python]生成器和yield关键字

生成器和yield关键字 1.生成器介绍: 概述: ​ 它指的是 generator, 类似于以前学过的: 列表推导式, 集合推导式, 字典推导式… 作用: ​ 降低资源消耗, 快速(批量)生成数据. 实现方式: ​ 1.推导式写法. my_generator = (i for i in range(5)) ​ 2.yield写法. def get_generator():for i

java基础总结11-面向对象7(super关键字)

在JAVA类中使用super来引用父类的成分,用this来引用当前对象,如果一个类从另外一个类继承,我们new这个子类的实例对象的时候,这个子类对象里面会有一个父类对象。怎么去引用里面的父类对象呢?使用super来引用,this指的是当前对象的引用,super是当前对象里面的父对象的引用。 1 super关键字测试 package cn.galc.test;/*** 父类* @autho

java基础总结08-面向对象4(static关键字)

原来一个类里面的成员变量,每new一个对象,这个对象就有一份自己的成员变量,因为这些成员变量都不是静态成员变量。对于static成员变量来说,这个成员变量只有一份,而且这一份是这个类所有的对象共享。 静态成员变量与非静态成员变量的区别 以下面的例子为例说明 package cn.galc.test;public class Cat {/*** 静态成员变量*/private static

java基础总结07-面向对象3(this关键字)

this是一个引用,它指向自身的这个对象。 看内存分析图 假设我们在堆内存new了一个对象,在这个对象里面你想象着他有一个引用this,this指向这个对象自己,所以这就是this,这个new出来的对象名字是什么,我们不知道,不知道也没关系,因为这并不影响这个对象在内存里面的存在,这个对象只要在内存中存在,他就一定有一个引用this。 看下面的例子分析: package cn.ga

关键字volatile有什么含意?

1. 并行设备的硬件寄存器。存储器映射的硬件寄存器通常加volatile,因为寄存器随时可以被外设硬件修改。当声明指向设备寄存器的指针时一定要用volatile,它会告诉编译器不要对存储在这个地 址的数据进行假设,也就是要去相应的内存地址里取。 2. 一个中断服务程序中修改的供其他程序检测的变量。volatile提醒编译器,它后面所定义的变量随时都有可能改变。因此编译后的程序每次需要存储或读取这