相信大多数Java程序员都学习过volatile这个关键字的用法。百度百科上对volatile的定义:
volatile是一个类型修饰符(type specifier),被设计用来修饰被不同线程访问和修改的变量。volatile的作用是作为指令关键字,确保本条指令不会因编译器的优化而省略,且要求每次直接读值。
可能有很多刚学Java的朋友们看了上面这段非常笼统的描述后仍然觉得云里雾里的。
下面我们就用一个具体的例子来学习volatile的用法。
看这个例子:
public class ThreadVerify {
public static Boolean stop = false;
public static void main(String args[]) throws InterruptedException {
Thread testThread = new Thread(){
@Override
public void run(){
int i = 1;
while(!stop){
//System.out.println("in thread: " + Thread.currentThread() + " i: " + i);
i++;
}
System.out.println("Thread stop i="+ i);
}
}
;
testThread.start();
Thread.sleep(1000);
stop = true;
System.out.println("now, in main thread stop is: " + stop);
testThread.join();
}
}
这段代码在主线程的第二行定义了一个布尔变量stop, 然后主线程启动一个新线程,在线程里不停得增加计数器i的值,直到主线程的布尔变量stop被主线程置为true才结束循环。
主线程用Thread.sleep停顿1秒后将布尔值stop置为true。
因此,我们期望的结果是,上述Java代码执行1秒钟后停止,并且打印出1秒钟内计数器i的实际值。
然而,执行这个Java应用后,你发现它进入了死循环,在任务管理器里发现这个Java程序CPU占用率飙升。
原因是什么呢?让我们温习下计算机专业课操作系统中讲过的内存模型的知识。
以Java内存模型为例,Java内存模型分为主内存(main memory)和工作内存(work memory)。主内存内的变量由所有线程共享,每个线程拥有自己的工作内存,里面的变量包含了线程局部变量。主内存中的变量如果被线程使用到,则线程的工作内存会维护一份主内存变量的副本拷贝。
线程对变量的所有读写操作必须在工作内存中进行,不能直接操作主内存中的变量。不同线程之间也无法直接访问对方的工作内存。线程间变量的传递需通过主内存来完成。线程、主内存、工作内存三者之间的交互关系如下图:
如果线程在自己的执行代码里修改了定义在主线程(主内存)中的变量,修改直接发生在线程的工作内存里,然后在某个时刻(Java程序员无法控制这个时刻,而是由JVM调度的),这个修改从工作内存写回到主内存。
回到我们的例子。尽管主线程修改了stop变量,但是仅仅修改了主内存中的值,而操作计数器的线程的工作内存里的stop变量还是旧的值,始终为false。因此这个线程陷入了死循环。
知道了原理,解决方案就很简单了。在stop变量前加上关键字volatile进行修饰,这样在计数器线程里每次读取stop的值时,volatile会强制该线程从主内存读取,而不是从当前线程的工作内存读取。这样就避免了死循环。下图显示1秒钟之后,计数器执行了14亿次。
要获取更多Jerry的原创技术文章,请关注公众号"汪子熙"或者扫描下面二维码:
相关推荐
volatile关键字可以保证可见性,即当一个线程修改了某个变量时,其他所有线程都知道该变量被修改了。这是因为volatile可以保证现在在读取volatile变量时,线程读取到的值是准确的。但是这并不意味着对volatile变量的...
在Java中,Volatile关键字是一个非常重要的概念,它与Java内存模型中的可见性、原子性和有序性息息相关。可见性是指线程之间的可见性,一个线程修改的状态对另一个线程是可见的。Volatile关键字可以确保多个线程之间...
Java中的`volatile`关键字是多线程编程中的一个重要概念,它的主要作用是解决并发环境下的可见性和有序性问题。在Java内存模型(JMM)中,每个线程都有自己的工作内存,其中包含了线程对共享变量的副本。线程间的...
`volatile`关键字是C++和Java等编程语言中用于处理多线程环境或者与硬件交互时的一个关键特性。它主要用于修饰变量,表明该变量的值可能会在编译器不知情的情况下发生变化,例如由其他线程修改、外部硬件事件影响...
Java中的`volatile`关键字是一个非常重要的并发编程工具,它的作用主要体现在两个方面:**可见性**和**有序性**。本文将深入解析`volatile`的关键字特性及其在实际编程中的应用。 1. 可见性: 当一个共享变量被`...
Java中的`volatile`关键字是并发编程中至关重要的一个特性,它用于解决多线程环境下的可见性和有序性问题。在Java内存模型(JMM)中,`volatile`关键字确保了共享变量在多线程间的正确通信。 首先,我们来看`...
Java中的`synchronized`关键字是多线程编程中的一个重要概念,用于控制并发访问共享资源时的同步机制。在Java中,当多个线程试图同时访问和修改同一块代码或数据时,可能会导致数据不一致的问题。为了解决这个问题,...
Java中的`volatile`关键字是用来解决多线程环境下的可见性和有序性问题的。它确保了被`volatile`修饰的变量在任何线程中都具有最新的值,但并不保证线程安全,即不保证并发操作的原子性。 首先,我们要理解Java内存...
Java中的`volatile`关键字是一个非常重要的工具,用于处理多线程环境中的共享变量。它确保了线程之间的数据可见性,但不提供原子性保证。理解`volatile`的关键在于掌握其两大特性:可见性和无锁。 **1. 可见性** `...
在Java编程语言中,`volatile`关键字是一个非常重要的线程安全特性,它的主要作用在于确保共享变量在多线程环境下的可见性和有序性。当一个变量被声明为`volatile`时,它有以下两个核心特性: 1. **可见性**:当一...
在Java编程语言中,`volatile`关键字是一个非常重要的同步机制,用于解决多线程环境下的可见性和有序性问题。在Java 5之前,由于内存模型的限制,`volatile`的效果并不理想,但在Java 5及以后版本中,JVM(Java...
当一个变量被`volatile`修饰后,例如`volatile int i`,如果在一个线程中修改了`i`的值,其他线程能够立刻看到这个变化。这是因为`volatile`变量的修改会强制将最新值写回主内存,并通知其他线程刷新它们的本地副本...
在Java编程语言中,`transient`关键字是一个非常重要的概念,主要用于对象序列化过程中对特定变量的控制。当一个类实现了`Serializable`接口时,其对象可以被序列化为一个持久化的存储形式,以便在网络上传输或存储...
在Java多线程编程中,volatile关键字扮演着重要的角色,它确保了被声明为volatile的变量在多个线程之间的可见性。然而,尽管volatile能够提供一定程度的线程安全,但并不能保证所有操作都是原子性的。这正是为什么在...
Java中的volatile关键字是一个关键的同步机制,它在多线程编程中扮演着重要的角色。在面试和技术讨论中,volatile经常成为焦点,但其工作原理却常常引发争议。本文将从JVM、C++以及汇编语言的角度深入探讨volatile的...
Java中的`volatile`关键字是用来处理多线程并发访问共享变量时的一种同步机制。它确保了被`volatile`修饰的变量的可见性和有序性,但不能保证原子性。 **可见性**: 当一个线程修改了`volatile`变量的值,其他线程...
Java中的`volatile`关键字是多线程编程中一个重要的概念,它主要解决了两个核心问题:可见性和有序性。在Java内存模型(JMM)中,`volatile`关键字确保了线程之间的通信更加有效和安全。 **一、防止指令重排序** ...
- **偏向锁**:默认情况下,Java 对象不带锁,当第一个线程访问同步块时,对象由无锁状态变为偏向锁状态。此时该线程拥有锁。 - **轻量级锁**:如果持有偏向锁的线程再次请求锁,或者有其他线程尝试获取该锁,那么...
在示例代码中,我们首先定义了一个非volatile变量flag,并在主线程中将其设置为true。在子线程中,我们使用while循环来检测flag的值。如果flag为false,则子线程将一直循环下去。然而,在实际运行中,我们发现子线程...
注释是 Java 程序中的一个重要组成部分,它们可以提高程序的可读性,易于区分代码行与注释行。Java 中有三种类型的注释:单行注释、多行注释和文档注释。 ### 单行注释 单行注释用 // 表示,编译器看到 // 会忽略...