volatile关键词在多线程中的应用

简介: volatile关键词在多线程中的应用

volatile关键词在多线程中的应用

volatile 是什么

首先我们就来介绍一下 volatile,它是 Java 中的一个关键字,是一种同步机制。当某个变量是共享变量,且这个变量是被 volatile 修饰的,那么在修改了这个变量的值之后,再读取该变量的值时,可以保证获取到的是修改后的最新的值,而不是过期的值。

// 摘自《Java 并发编程 78 讲》

使用场景

基本属性的赋值

例如int,boolean的赋值,因为赋值操作为原子操作。应要注意到i++ 不是原子操作,不在此列。而布尔值的操作基本上是直接进行赋值的,故《Java 并发编程 78 讲》对使用场景的介绍指“布尔标记位”。这种场景不做深究。

作为触发器

先看一段代码,以及结果:

public class VolatileClass {
    int beforeVolatile = 0;
    /**
     * 作为触发器
     */
    volatile boolean flag;
    int afterVolatile = 0;
    public VolatileClass() {
        flag = false;
    }
    public void init() {
        beforeVolatile = 1;
        flag = true;
        try {
            Thread.sleep(50);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        afterVolatile = 2;
        System.out.println(String.format("程序初始化结束:beforeVolatile[%d],flag[%b],afterVolatile[%d]",
                beforeVolatile, flag, afterVolatile));
    }
    public void afterInit() {
        while (!flag) {
            // todo sout << "程序还未完成初始化";
            System.out.println(String.format("程序还未完成初始化:beforeVolatile[%d],flag[%b],afterVolatile[%d]",
                    beforeVolatile, flag, afterVolatile));
        }
        System.out.println(String.format("检测程序初始化结束:beforeVolatile[%d],flag[%b],afterVolatile[%d]",
                beforeVolatile, flag, afterVolatile));
    }
    public static void main(String[] args) {
        VolatileClass volatileClass = new VolatileClass();
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                volatileClass.afterInit();
            }
        });
        thread.start();
        Thread init = new Thread(new Runnable() {
            @Override
            public void run() {
                volatileClass.init();
            }
        });
        init.start();
        try {
            Thread.sleep(150);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

结果:

程序还未完成初始化:beforeVolatile[0],flag[false],afterVolatile[0]
检测程序初始化结束:beforeVolatile[1],flag[true],afterVolatile[0]
程序初始化结束:beforeVolatile[1],flag[true],afterVolatile[2]
Process finished with exit code 0

可以观察到几个现象:

  • 线程thread是先启动的,他一开始进入while时,flag为false,初始化未完成,而当init线程去修改flag为true之后,thread线程立马就跳出了while循环,打印了“检测程序初始化结束”,触发器生效!
  • 在“检测程序初始化结束”中打印的beforeVolatile值已经为1,这根据"happens before"规则,是可以保证发生在volatile 修饰的flag之前修改的变量也具有了可见性。

volatile的作用域

虽然基本上用作触发器是用boolean变量,但是其作用于对象、数组。那么其对象的属性、数组的元素发生变化,都会被此关键词保证其可见性,此处可以通过修改flag变量的类型进行实验,已亲测,便不贴代码占文了。

volatile的原理

说一下自己的理解,并不是官方解释。

首先看一下JMM(JAVA内存模型)

每个线程都有他的本地内存,是从共享内存中拷贝出来的,这也是造成线程之间数据不可见性的原因。当被volatile修饰的对象被修改,那么JMM保证它会被立马更新到共享内存中,同时其他线程会从共享内存中读取其值。

同时,个人理解(还未考证),在flush的这个过程,他并不是将volatile修饰的单个对象的修改刷入共享内存,而是将发生修改过的对象全部刷入共享内存,同时refresh过程会将共享内存中全部更新的对象全部载入到本地内存中。这样就形成了happens-before。

另外,volatile关键字会一定程度上禁止编译器在编译阶段优化时一定范围的指令重排序,以保证happens-before规则。有兴趣可以另外自行了解。

目录
相关文章
|
20天前
|
缓存 Java 开发者
Java多线程并发编程:同步机制与实践应用
本文深入探讨Java多线程中的同步机制,分析了多线程并发带来的数据不一致等问题,详细介绍了`synchronized`关键字、`ReentrantLock`显式锁及`ReentrantReadWriteLock`读写锁的应用,结合代码示例展示了如何有效解决竞态条件,提升程序性能与稳定性。
73 6
|
18天前
|
监控 Java 数据库连接
Java线程管理:守护线程与用户线程的区分与应用
在Java多线程编程中,线程可以分为守护线程(Daemon Thread)和用户线程(User Thread)。这两种线程在行为和用途上有着明显的区别,了解它们的差异对于编写高效、稳定的并发程序至关重要。
26 2
|
24天前
|
数据采集 存储 数据处理
Python中的多线程编程及其在数据处理中的应用
本文深入探讨了Python中多线程编程的概念、原理和实现方法,并详细介绍了其在数据处理领域的应用。通过对比单线程与多线程的性能差异,展示了多线程编程在提升程序运行效率方面的显著优势。文章还提供了实际案例,帮助读者更好地理解和掌握多线程编程技术。
|
24天前
|
存储 监控 安全
深入理解ThreadLocal:线程局部变量的机制与应用
在Java的多线程编程中,`ThreadLocal`变量提供了一种线程安全的解决方案,允许每个线程拥有自己的变量副本,从而避免了线程间的数据竞争。本文将深入探讨`ThreadLocal`的工作原理、使用方法以及在实际开发中的应用场景。
46 2
|
28天前
|
安全 Java 开发者
Java 多线程并发控制:深入理解与实战应用
《Java多线程并发控制:深入理解与实战应用》一书详细解析了Java多线程编程的核心概念、并发控制技术及其实战技巧,适合Java开发者深入学习和实践参考。
52 6
|
28天前
|
存储 安全 Java
Java多线程编程中的并发容器:深入解析与实战应用####
在本文中,我们将探讨Java多线程编程中的一个核心话题——并发容器。不同于传统单一线程环境下的数据结构,并发容器专为多线程场景设计,确保数据访问的线程安全性和高效性。我们将从基础概念出发,逐步深入到`java.util.concurrent`包下的核心并发容器实现,如`ConcurrentHashMap`、`CopyOnWriteArrayList`以及`BlockingQueue`等,通过实例代码演示其使用方法,并分析它们背后的设计原理与适用场景。无论你是Java并发编程的初学者还是希望深化理解的开发者,本文都将为你提供有价值的见解与实践指导。 --- ####
|
2月前
|
存储 并行计算 安全
C++多线程应用
【10月更文挑战第29天】C++ 中的多线程应用广泛,常见场景包括并行计算、网络编程中的并发服务器和图形用户界面(GUI)应用。通过多线程可以显著提升计算速度和响应能力。示例代码展示了如何使用 `pthread` 库创建和管理线程。注意事项包括数据同步与互斥、线程间通信和线程安全的类设计,以确保程序的正确性和稳定性。
|
2月前
|
监控 Java
在实际应用中选择线程异常捕获方法的考量
【10月更文挑战第15天】选择最适合的线程异常捕获方法需要综合考虑多种因素。没有一种方法是绝对最优的,需要根据具体情况进行权衡和选择。在实际应用中,还需要不断地实践和总结经验,以提高异常处理的效果和程序的稳定性。
30 3
|
2月前
|
数据采集 存储 Java
Crawler4j在多线程网页抓取中的应用
Crawler4j在多线程网页抓取中的应用
|
1月前
|
Java 开发者
Java中的多线程基础与应用
【10月更文挑战第24天】在Java的世界中,多线程是提高效率和实现并发处理的关键。本文将深入浅出地介绍如何在Java中创建和管理多线程,以及如何通过同步机制确保数据的安全性。我们将一起探索线程生命周期的奥秘,并通过实例学习如何优化多线程的性能。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开一扇通往高效编程的大门。
23 0