posts - 310, comments - 6939, trackbacks - 0, articles - 3
  BlogJava :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理

Think in Java 4th--Java中的volatile

Posted on 2007-09-11 14:31 诗特林 阅读(1708) 评论(2)  编辑  收藏 所属分类: Think In Java 4th
Think in Java 4th--Java中的volatile

我们知道,在Java中设置变量值的操作,除了long和double类型的变量外都是原子操作,也就是说,对于变量值的简单读写操作没有必要进行同步。

这在JVM 1.2之前,Java的内存模型实现总是从主存读取变量,是不需要进行特别的注意的。而随着JVM的成熟和优化,现在在多线程环境下volatile关键字的使用变得非常重要。

在当前的Java内存模型下,线程可以把变量保存在本地内存(比如机器的寄存器)中,而不是直接在主存中进行读写。这就可能造成一个线程在主存中修改了一个变量的值,而另外一个线程还继续使用它在寄存器中的变量值的拷贝,造成数据的不一致。

要解决这个问题,只需要像在本程序中的这样,把该变量声明为volatile(不稳定的)即可,这就指示JVM,这个变量是不稳定的,每次使用它都到主存中进行读取。一般说来,多任务环境下各任务间共享的标志都应该加volatile修饰。

Volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。 

Java语言规范中指出:为了获得最佳速度,允许线程保存共享成员变量的私有拷贝,而且只当线程进入或者离开同步代码块时才与共享成员变量的原始值对比。 

这样当多个线程同时与某个对象交互时,就必须要注意到要让线程及时的得到共享成员变量的变化。 

而volatile关键字就是提示VM:对于这个成员变量不能保存它的私有拷贝,而应直接与共享成员变量交互。 

使用建议:在两个或者更多的线程访问的成员变量上使用volatile。当要访问的变量已在synchronized代码块中,或者为常量时,不必使用。 

由于使用volatile屏蔽掉了VM中必要的代码优化,所以在效率上比较低,因此一定在必要时才使用此关键字。

请看如下示例:

 

 1package ch17_concurrency;
 2
 3
 4class UnresponsiveUI{
 5 private volatile double d=1;
 6 public UnresponsiveUI() throws Exception{
 7  while(d>0){
 8   d=d+(Math.PI+Math.E)/d;
 9  }

10  System.in.read();
11 }

12}

13
14public class _19_ResponsiveUI extends Thread {
15 private static volatile double d=1;
16 public _19_ResponsiveUI(){
17  setDaemon(true);
18  start();
19 }

20 public void run(){
21  while(true){
22   //System.out.println(d);
23   d=d+(Math.PI+Math.E)/d;
24  }

25 }

26
27 public static void main(String[] args) throws Exception {
28  // TODO Auto-generated method stub
29  //new UnresponsiveUI();
30  new _19_ResponsiveUI();
31  System.in.read();
32  System.out.println(d);
33 }

34
35}

36

评论

# re: Think in Java 4th--Java中的volatile  回复  更多评论   

2007-09-11 16:48 by dennis
volatile需要强调的一点就是,它仅仅保证多线程环境下的可见性,不保证操作的原子性,举的例子不大恰当吧,想更多了解还是看看这个帖子

http://www.javaeye.com/topic/109150

# re: Think in Java 4th--Java中的volatile[未登录]  回复  更多评论   

2008-05-23 14:26 by dd
简单点就是线程在运行的时候为了提高效率所以被允许偷一点点懒,即忽视多线程的可能,不更新主存中的原本(线程使用的值是副本)。

VOLATILE即告诉JVM,不能偷懒!

这个特性应该是从C移植过来的。而C是没有线程的,所以说,JAVA的线程其实做得很勉强。

因为从语义的角度讲,线程的这种偷懒工作方式应该是不被允许的。JVM首先应该保证正确性(主存跟工作区的同步),然后再考虑性能优化。不正确的东西,性能再好也是白搭。



只有注册用户登录后才能发表评论。


网站导航: