1.JUC线程高级-volatile关键字与内存可见性

简介: 1. 多线程操作共享变量引发问题我们通过下面代码来分析下内存可见性问题: package com.pyy.juc; public class TestVolatile { public static...

1. 多线程操作共享变量引发问题

我们通过下面代码来分析下内存可见性问题:

    package com.pyy.juc;
    
    public class TestVolatile {
    
        public static void main(String[] args) {
    
            // 这个线程为flag 修改值
            ThreadDemo td = new ThreadDemo();
            new Thread(td).start();
    
    
            // 主线程无线循环判断这个flag值
            while(true) {
                if(td.isFlag()) {
                    System.out.println("============");
                    break;
                }
            }
        }
    }
    
    class ThreadDemo implements Runnable {
        private boolean flag = false;
    
        @Override
        public void run() {
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
    
            flag = true;
    
            System.out.println("flag=" + isFlag());
        }
    
        public boolean isFlag() {
            return flag;
        }
    }

运行代码,会发现控制台输出:flag=true,但程序(主线程)并没有结束。这里就涉及到多线程的内存可见性问题:

img_102b9f201966ee25812f608b87a68ca9.png

出现上述问题的原因,主要是因为多个线程操作共享数据彼此不可见

解决内存可见性问题可以使用同步锁(synchronized)

        while(true) {
            synchronized(td) {
                if(td.isFlag()) {
                    System.out.println("============");
                    break;
                }
            }
        }

通过同步锁方式(可以保证每次都刷新主存,保证共享数据的同步性),但只要用到锁就会引发线程等待,影响代码效率。

2. volatile关键字

这里就引出了一个新的关键字volatile,它就可以保证多个线程操作共享数据内存中的数据是彼此可见的。底层是通过内存屏障,我们可以理解volatile修饰的变量是在主存中的数据直接操作。

读写变量都在 主存中直接操作,进而保证多线程可见性。而且效率上要比同步锁高的多。

img_15f4a90abe8fe6504f9697a474eca37d.png

其实有说volatile效率底,是因为jvm底层有个指令排序,但使用volatile修饰的变量会有一个禁止指令重排限制

3. 代码重构后

    ...
    // 使用volatile修饰变量
    private volatile boolean flag = false;
    ...

就可以解决多个线程操作共享数据彼此不可见问题

4. volatile和synchronized区别

volatile只是一个相比synchronized来说较为轻量级的同步策略

  1. volatile 不具备互斥性
  2. volatile 不能保证修饰变量的原子性

后面我们在谈论下有关变量原子性和CAS算法问题

目录
相关文章
|
2月前
|
存储 SQL 缓存
揭秘Java并发核心:深度剖析Java内存模型(JMM)与Volatile关键字的魔法底层,让你的多线程应用无懈可击
【8月更文挑战第4天】Java内存模型(JMM)是Java并发的核心,定义了多线程环境中变量的访问规则,确保原子性、可见性和有序性。JMM区分了主内存与工作内存,以提高性能但可能引入可见性问题。Volatile关键字确保变量的可见性和有序性,其作用于读写操作中插入内存屏障,避免缓存一致性问题。例如,在DCL单例模式中使用Volatile确保实例化过程的可见性。Volatile依赖内存屏障和缓存一致性协议,但不保证原子性,需与其他同步机制配合使用以构建安全的并发程序。
61 0
|
3月前
|
安全 Java 开发者
探索Java内存模型:可见性、有序性和并发
在Java的并发编程领域中,内存模型扮演了至关重要的角色。本文旨在深入探讨Java内存模型的核心概念,包括可见性、有序性和它们对并发实践的影响。我们将通过具体示例和底层原理分析,揭示这些概念如何协同工作以确保跨线程操作的正确性,并指导开发者编写高效且线程安全的代码。
|
3月前
|
缓存 安全 Java
Java面试题:解释volatile关键字的作用,以及它如何保证内存的可见性
Java面试题:解释volatile关键字的作用,以及它如何保证内存的可见性
63 4
|
3月前
|
安全 Java
Java面试题:解释synchronized关键字在Java内存模型中的语义
Java面试题:解释synchronized关键字在Java内存模型中的语义
41 1
|
3月前
|
存储 缓存 Java
(一) 玩命死磕Java内存模型(JMM)与 Volatile关键字底层原理
文章的阐述思路为:先阐述`JVM`内存模型、硬件与`OS`(操作系统)内存区域架构、`Java`多线程原理以及`Java`内存模型`JMM`之间的关联关系后,再对`Java`内存模型进行进一步剖析,毕竟许多小伙伴很容易将`Java`内存模型(`JMM`)和`JVM`内存模型的概念相互混淆,本文的目的就是帮助各位彻底理解`JMM`内存模型。
|
3月前
|
微服务
多线程内存模型问题之在单例模式中,volatile关键字的作用是什么
多线程内存模型问题之在单例模式中,volatile关键字的作用是什么
|
3月前
线程可见性和关键字volatile
线程可见性和关键字volatile
|
3月前
|
存储 缓存 安全
Java面试题:介绍一下jvm中的内存模型?说明volatile关键字的作用,以及它如何保证可见性和有序性。
Java面试题:介绍一下jvm中的内存模型?说明volatile关键字的作用,以及它如何保证可见性和有序性。
30 0
|
2月前
|
存储 编译器 C语言
【C语言篇】数据在内存中的存储(超详细)
浮点数就采⽤下⾯的规则表⽰,即指数E的真实值加上127(或1023),再将有效数字M去掉整数部分的1。
|
3月前
|
存储 分布式计算 Hadoop
HadoopCPU、内存、存储限制
【7月更文挑战第13天】
215 14