0 0

谁能真正整明白java volatile 关键字? 一知半解的莫入!0

/**
 *  VolatileTest.java
 *
 *  Copyright ekupeng,Inc. 2012   
 */
package test;

/**
 * @ClassName: VolatileTest
 * @Description: Volatile测试
 * @author Emerson emsn1026@gmail.com
 * @date 2012-11-29 下午06:57:44
 * @version V1.0
 * 
 */
public class VolatileTest extends Thread {

	// 非volatile标志
	private static boolean flag1 = false;
	// volatile标志
	private static volatile boolean flag2 = false;

	private int i = 0;

	public void run() {
		//Object o = new Object();

		//synchronized (o) {
			/*
			 * 注释1
			 */
			while (!flag1) {
				i++;
                                //注意 : System.out.println(i);
				/*
				 * 注释2
				 */
				if (flag2) {
					System.out.println("over:" + i);
					break;
				}
			}
		//}
	}

	public static void main(String[] args) {

		VolatileTest t = new VolatileTest();
		t.start();

		try {
			Thread.currentThread().sleep(2000);
			// 先更改flag1
			t.flag1 = true;
			/*
			 * 注释3
			 */
			Thread.currentThread().sleep(1000);
			// 将flag2置为true,如果有机会进入if(flag2),则将退出循环
			t.flag2 = true;
		} catch (InterruptedException e) {
			e.printStackTrace();
		}

	}

}



因为预览时发现大段的注释会自动换行,影响阅读代码,所以我将代码中的注释提取出来:
注释1
外围标志flag1为非volatile,该线程(t)跑起来后由另一线程(main)将flag1改为true后,如果出现情况1.flag1如果不从主存重新读取,那他将继续以false运行,所以会继续循环并进入内部的flag2的if判断;如果出现情况2.flag1从主存重新读取,那他将以true运行,所以会跳出循环,也就没有机会进入flag2的if判断了;
注释2
如果出现情况1,将进入该判断,内部标志flag2为volatile,当线程(main)将flag2改为true后,因为flag2会从主存重新读取,将以true运行,所以将跳出循环,并打印"over"语句
注释3
为了确保flag1的变更有机会被t察觉,并保证flag2能在flag1变为true后进行一次以上while(!flag1)条件判断后再判断if(flag2),sleep1秒(1秒可以跑很多循环了)

以上是我为了说明volatile的功能写的一段程序,目的是想说一个线程1在循环中通过非volatile的布尔变量来进行条件判断,即使在另一个线程2中修改了该布尔变量,由于该线程1的代码执行得到了某种性能优化,不会从主存重新读取布尔值,导致进入死循环,直到内部的volatile布尔值被改变才跳出。

我的问题是:原本我以为会像预想那样的输出“over”语句,这样也说明了volatile的用处。但是我尝试了Sun JDK1.6,1.5,1.4,1.3,1.2(因为volatile是针对jit带来的优化,所以1.2之前的版本就没有尝试)之后发现只有1.2下才会看到该程序对于volatile的演示效果,输出了“over”语句。其他的都只是在外围的while(!flag1)中实时察觉flag1的变化并跳出了循环。原本以为是hotspot的问题,但是我尝试了hotspot的server或client,以及1.3的classic,都是没有效果的,只有1.2才能看到volatile的演示效果。哪位大神给细说下这个情况?

ps:Object那把锁没有实质的意义,只是进出synchronized块时会重新从主存同步数据,我当时随手写了测了下,所以大家可以不考虑,我暂且将它注掉吧....为了说明这一点,我加了行代码:
//注意 : System.out.println(i);
这行代码要是去掉注释,volatile在JDK1.2下也将失去作用,因为System.out.println中含有同步块,一执行该方法,变量将从主存中重新读取。
2013年10月10日 17:15

7个答案 按时间排序 按投票排序

0 0

采纳的答案

以上是我为了说明volatile的功能写的一段程序,目的是想说一个线程1在循环中通过非volatile的布尔变量来进行条件判断,即使在另一个线程2中修改了该布尔变量,由于该线程1的代码执行得到了某种性能优化,不会从主存重新读取布尔值,导致进入死循环,直到内部的volatile布尔值被改变才跳出。
你可能对JMM的理解有偏差。JMM只会说确保一些happen-before。但是对那些非happen-before的情况也不会出现你这种极端情况吧。


即使在另一个线程2中修改了该布尔变量,由于该线程1的代码执行得到了某种性能优化,不会从主存重新读取布尔值,导致进入死循环。

多线程编程的问题为什么难以发现,是因为那些非happen-before的代码在大多数情况表现的都是happen-before。如果按你这么说,那些不符合happen-before的问题立马就会出现那就好了。

2013年10月11日 17:15
0 0

http://www.oldcaptain.cc/articles/2013/08/21/1377092100971.html  自己瞅瞅吧

2013年10月15日 17:36
0 0

重要的一点:原子操作

2013年10月12日 11:46
0 0

volatile保证一个线程对变量修改之后其它线程立即可见
如果不用volatile修改,所有对象成员属性发生的修改只有在线程结束后其它线程才能发现这个修改

但是volatile不保证线程安全,所以对volatile变量执行++ -- += -= *= /= 和其它运算后再赋值的操作,可能会出现数据不一致现象。

2013年10月12日 10:49
0 0

JMM会确保volatile的改变对别的线程立马可见。
但是JMM又没确保非volatile的改变不会对别的线程立马可见。
所以我没看懂

   t.flag1 = true;  
            /* 
             * 注释3 
             */  
            Thread.currentThread().sleep(1000);  

这一段的用意。
你这么一用那么另外个线程怎么可能不会对flag1的变化不可见呢。都睡了1S了。也该让别的线程察觉了吧。那循环都跳出了。怎么会进入到flag2的循环呢。。。

2013年10月11日 17:08
0 0

主内存和工作内存的区别,当对变量做非原子性操作时,线程修改了工作内存中的变量内容并不会立刻把结果同步到主内存中,如果加了此关键字就会立刻同步。

2013年10月11日 16:43
0 0

关键字:主内存,工作内存

2013年10月11日 10:01

相关推荐

    java volatile 关键字实战

    java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java ...

    java volatile 关键字 学习

    java volatile 关键字 学习

    Java并发编程:volatile关键字解析

    ### Java并发编程:volatile关键字解析 #### 一、内存模型的相关概念 在深入了解`volatile`关键字之前,我们首先需要理解计算机内存模型的一些基本概念。在现代计算机系统中,CPU为了提高执行效率,会将频繁访问的...

    java里的volatile关键字详解

    "Java中的Volatile关键字详解" Java中的Volatile关键字详解是Java中的一种关键字,用于保证线程之间的可见性、原子性和有序性。下面是对Java中的Volatile关键字详解的知识点总结: 一、基本概念 1. 可见性:可见...

    深入解析Java中的volatile关键字:原理、应用与实践

    在Java并发编程中,volatile关键字是一种轻量级的同步机制,它用于确保变量的可见性和有序性。本文将详细探讨volatile关键字的工作原理、使用场景以及如何在实际开发中正确使用volatile。 volatile关键字是Java并发...

    Java线程:volatile关键字

    Java 线程 volatile 关键字详解 Java™ 语言包含两种内在的同步机制:同步块(或方法)和 volatile 变量。volatile 变量的同步性较差,但它有时更简单并且开销更低。volatile 变量可以被看作是一种 “程度较轻的 ...

    Java volatile关键字详解.docx

    volatile

    Java并发volatile关键字.docx

    Java并发编程中,volatile关键字扮演着重要的角色,它是一种轻量级的同步机制,与synchronized相比,volatile在性能上更优,因为它不会导致线程阻塞。在深入理解volatile的关键特性之前,我们需要先了解Java内存模型...

    一文精通Java中的volatile关键字

    Java中的`volatile`关键字是多线程编程中的一个重要概念,它的主要作用是确保共享变量的可见性和禁止指令重排序。本文将深入探讨`volatile`的关键特性、工作原理以及使用注意事项。 1. 可见性: `volatile`关键字...

    详解C中volatile关键字

    详解C中volatile关键字 volatile关键字是C语言中一个重要的关键字,它对变量的声明在不同编译环境下可能造成不同的结果。volatile关键字的作用是提醒编译器它后面所定义的变量随时都有可能改变,因此编译后的程序...

    java语言的volatile教程,java语言的volatile关键字是干什么用的

    java语言的volatile教程,java语言的volatile关键字到底怎么用

    java volatile关键字使用方法及注意事项

    volatile关键字在Java编程语言中扮演着重要的角色,特别是在多线程环境下的同步和可见性问题。它是Java内存模型(JMM)的一部分,用于确保共享变量的可见性和有序性,但不保证原子性。 1. **volatile的可见性**:当...

    Java Volatile关键字同步机制详解

    Java Volatile关键字同步机制详解 Java Volatile关键字是Java语言中的一种同步机制,它可以保证在多线程环境下变量的可见性和原子性。通过使用Volatile关键字,可以确保在多线程环境下对变量的修改是可见的,并且...

    java里的volatile关键字详解.pdf

    java里的volatile关键字详解.pdf

    Java中volatile关键字实现原理

    Java中volatile关键字实现原理 volatile关键字是Java语言中的一种机制,用于保证变量在多线程之间的可见性。它是Java.util.concurrent包的核心,没有volatile就没有那么多的并发类供我们使用。本文详细解读一下...

    Java——volatile关键字详解

    Java中的`volatile`关键字是一个非常重要的并发控制工具,它提供了比`synchronized`关键字更为轻量级的同步机制。`volatile`关键字的主要作用是确保多线程环境下的可见性和禁止指令重排序,但不保证原子性。 **...

    4.4 volatile关键字有什么作用?.rar

    在IT行业中,volatile关键字在编程,尤其是嵌入式系统和多线程编程中扮演着重要的角色。本篇文章将深入探讨volatile关键字的作用,并结合GD32F303单片机的使用场景,来阐述它在实际开发中的应用。 首先,volatile...

    深入理解 volatile 关键字.doc

    volatile关键字是Java语言的高级特性,它可以保证可见性和禁止指令重排序,但是要弄清楚其工作原理,需要先弄懂Java内存模型。 保证可见性 volatile关键字可以保证可见性,即当一个线程修改了某个变量时,其他所有...

    详解Java面试官最爱问的volatile关键字

    "Java面试官最爱问的volatile关键字详解" Java面试官最爱问的volatile关键字是Java并发编程中一个重要的概念,了解volatile关键字可以帮助开发者更好地理解Java内存模型(JMM)和Java并发编程的特性。本文将详细...

    Java中volatile关键字的总结.docx

    Java中的`volatile`关键字是用于处理并发编程中线程间通信和可见性的重要工具。它是一种轻量级的同步机制,比`synchronized`锁更为高效,因为它不会导致线程上下文的切换和调度,减少了不必要的开销。然而,`...

Global site tag (gtag.js) - Google Analytics