Volatile 变量

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

注意:

假如有两个线程共享volatile变量,及其他普通变量时,无论某volatile变量是否发生改变,线程B在读取该volatile变量时,不仅能看到其他线程A对该volatile变量的写入(如果存在)操作,A线程之前所有可见的共享变量,在B线程读同一个volatile变量后,都将立即变得对B线程可见

 

public class Volatile extends Object implements Runnable {  
    //value变量没有被标记为volatile  
    private int value;    
    //missedIt变量被标记为volatile  
    private volatile boolean missedIt;  
    //creationTime不需要声明为volatile,因为代码执行中它没有发生变化  
    private long creationTime;   
  
    public Volatile() {  
        value = 10;  
        missedIt = false;  
        //获取当前时间,亦即调用Volatile构造函数时的时间  
        creationTime = System.currentTimeMillis();  
    }  
  
    public void run() {  
        print("entering run()");  
  
        //循环检查value的值是否不同  
        while ( value < 20 ) {
            //如果missedIt的值被修改为true,则通过break退出循环  
        	 //*****************线程读取volatile变量missedIt时,同步更新了value的值,所以workMethod改变value值后,线程退出while循环,如果将missedIt改为false那么循环会一直进行下去*************************//
            if  ( missedIt ) { 
                //进入同步代码块前,将value的值赋给currValue  
                int currValue = value;  
                //在一个任意对象上执行同步语句,目的是为了让该线程在进入和离开同步代码块时,  
                //将该线程中的所有变量的私有拷贝与共享内存中的原始值进行比较,  
                //从而发现没有用volatile标记的变量所发生的变化  
                Object lock = new Object();  
                synchronized ( lock ) {  
                    //不做任何事  
                }  
                //离开同步代码块后,将此时value的值赋给valueAfterSync  
                int valueAfterSync = value;  
                print("in run() - see value=" + currValue +", but rumor has it that it changed!");  
                print("in run() - valueAfterSync=" + valueAfterSync);  
                break;   
            }  
        }
        print("leaving run()");  
    }  
  
    public void workMethod() throws InterruptedException {  
        print("entering workMethod()");  
        print("in workMethod() - about to sleep for 2 seconds");  
        Thread.sleep(2000);  
        //仅在此改变value的值  
        value = 50;  
        print("in workMethod() - just set value=" + value);  
        print("in workMethod() - about to sleep for 5 seconds");  
        Thread.sleep(5000);  
        //仅在此改变missedIt的值  
//        missedIt = true;  
        print("in workMethod() - just set missedIt=" + missedIt);  
        print("in workMethod() - about to sleep for 3 seconds");  
        Thread.sleep(3000);  
        print("leaving workMethod()");  
    }  
  
    public static void main(String[] args) {  
        try {  
            //通过该构造函数可以获取实时时钟的当前时间  
            Volatile vol = new Volatile();  
  
            //稍停100ms,以让实时时钟稍稍超前获取时间,使print()中创建的消息打印的时间值大于0  
            Thread.sleep(100);    
  
            Thread t = new Thread(vol);  
            t.start();  
  
            //休眠100ms,让刚刚启动的线程有时间运行  
//            Thread.sleep(100);    
            //workMethod方法在main线程中运行  
            vol.workMethod();
        } catch ( InterruptedException x ) {  
            System.err.println("one of the sleeps was interrupted");  
        }  
    }  
    
    /* 
    *该方法的功能是在要打印的msg信息前打印出程序执行到此所化去的时间,以及打印msg的代码所在的线程 
    */  
        private void print(String msg) {  
            //使用java.text包的功能,可以简化这个方法,但是这里没有利用这一点  
            long interval = System.currentTimeMillis() - creationTime;  
            String tmpStr = "    " + ( interval / 1000.0 ) + "000";       
            int pos = tmpStr.indexOf(".");  
            String secStr = tmpStr.substring(pos - 2, pos + 4);  
            String nameStr = "        " + Thread.currentThread().getName();  
            nameStr = nameStr.substring(nameStr.length() - 8, nameStr.length());      
            System.out.println(secStr + " " + nameStr + ": " + msg);  
        }  
}  

 

  

 

posted @ 2017-08-27 21:34  猎空  阅读(275)  评论(0编辑  收藏  举报