java 里面volatile什么意思啊

来源:互联网 发布:java自定义表单引擎 编辑:程序博客网 时间:2024/04/30 10:33

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

 

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

 

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

 

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

 

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

 

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

 

 

一般的,如果多个线程协作存、取某个变量时,一般需要用到synchronized关键字进行同步操作,如:

public class MyTestThread extends MyTest implements Runnable {

    private boolean _done = false;

    public synchronized boolean getDone()

    {

       return _done;

    }

    public synchronized void setDone(boolean b)

    {

       _done = b;

    }

   

    public void run( ) {

       boolean done;

       done = getDone();

       while (!done) {

           repaint( );

           try {

               Thread.sleep(100);

           } catch (InterruptedException ie) {

               return;

           }

       }

    }

}

或者:

public class MyTestThread extends MyTest implements Runnable {

    private boolean _done = false;

    public void setDone(boolean b)

    {

       synchronized(this)

       {

           _done = b;

       }

    }

   

    public void run( ) {

       boolean done;

       synchronized(this)

       {

           done = _done;

       }

       

       while (!done) {

           repaint( );

           try {

               Thread.sleep(100);

           } catch (InterruptedException ie) {

               return;

           }

       }

    }

}

但是,通过volatile关键字,我们可以大大简化:

public class MyTestThread extends MyTest implements Runnable {

    private volatile boolean done = false;

    public void run( ) {

       while (!done) {

           repaint( );

            try {

               Thread.sleep(100);

           } catch (InterruptedException ie) {

               return;

           }

       }

    }

    public void setDone(boolean b) {

       done = b;

    }

}

原创粉丝点击