怎样去理解Java中的volatile

volatile修饰符修饰变量,不可以修饰类和方法,volatile不是线程安全的,但是一写多读的多线程情况下是线程安全的。

 

多线程的特性:

  •   可见性  各线程之间的变量会同步到主线程
  •   原子性

volatile是可见的,但不是原子的。

synchronized是可见的也是原子的。

 

一个线程写,多个线程读可以使用volatile,这样的情况下的多线程是安全的。

多个线程写,多个线程读不能使用volatile,这样的情况下的多线程是不安全的。

package crelle.test.juc;

/**
 * 一个线程写,多个线程读 线程安全
 */
public class VolatileSingleWriterTest {
    private static volatile int MY_INT = 0;
    public static void main(String[] args) {
        new ChangeListener().start();
        new ChangeListener().start();
        new ChangeListener().start();
        new ChangeListener().start();
        new ChangeListener().start();
        new ChangeListener().start();
        new ChangeMaker().start();
    }
    static class ChangeListener extends Thread {
        @Override
        public void run() {
            int local_value = MY_INT;
            while ( local_value < 5){
                if( local_value!= MY_INT){
                    System.out.println("Got Change for MY_INT : "+ MY_INT);
                    local_value= MY_INT;
                }
            }
        }
    }

    static class ChangeMaker extends Thread{
        @Override
        public void run() {
            int local_value = MY_INT;
            while (MY_INT <5){
                System.out.println("Incrementing MY_INT to : "+( local_value+1));
                MY_INT = ++local_value;
                try {
                    Thread.sleep(500);
                } catch (InterruptedException e) { e.printStackTrace(); }
            }
        }
    }
}
结果:
Incrementing MY_INT to : 1
Got Change for MY_INT : 1
Got Change for MY_INT : 1
Got Change for MY_INT : 1
Got Change for MY_INT : 1
Got Change for MY_INT : 1
Got Change for MY_INT : 1
Incrementing MY_INT to : 2
Got Change for MY_INT : 2
Got Change for MY_INT : 2
Got Change for MY_INT : 2
Got Change for MY_INT : 2
Got Change for MY_INT : 2
Got Change for MY_INT : 2
Incrementing MY_INT to : 3
Got Change for MY_INT : 3
Got Change for MY_INT : 3
Got Change for MY_INT : 3
Got Change for MY_INT : 3
Got Change for MY_INT : 3
Got Change for MY_INT : 3
Incrementing MY_INT to : 4
Got Change for MY_INT : 4
Got Change for MY_INT : 4
Got Change for MY_INT : 4
Got Change for MY_INT : 4
Got Change for MY_INT : 4
Got Change for MY_INT : 4
Incrementing MY_INT to : 5
Got Change for MY_INT : 5
Got Change for MY_INT : 5
Got Change for MY_INT : 5
Got Change for MY_INT : 5
Got Change for MY_INT : 5
Got Change for MY_INT : 5

Process finished with exit code 0

  

package crelle.test.juc;

/**
 * 多个线程写,多个线程读 线程不安全
 */
public class VolatileMultiWriterTest {
    private static volatile int MY_INT = 0;
    public static void main(String[] args) {
        new ChangeListener().start();
        new ChangeMaker().start();
        new ChangeMaker().start();
        new ChangeMaker().start();
        new ChangeMaker().start();
        new ChangeMaker().start();
        new ChangeMaker().start();
    }
    static class ChangeListener extends Thread {
        @Override
        public void run() {
            int local_value = MY_INT;
            while ( local_value < 5){
                if( local_value!= MY_INT){
                    System.out.println("Got Change for MY_INT : "+ MY_INT);
                    local_value= MY_INT;
                }
            }
        }
    }

    static class ChangeMaker extends Thread{
        @Override
        public void run() {
            int local_value = MY_INT;
            while (MY_INT <5){
                System.out.println("Incrementing MY_INT to : "+( local_value+1));
                MY_INT = ++local_value;
                try {
                    Thread.sleep(500);
                } catch (InterruptedException e) { e.printStackTrace(); }
            }
        }
    }
}
结果:
Incrementing MY_INT to : 1 Incrementing MY_INT to : 1 Incrementing MY_INT to : 1 Got Change for MY_INT : 1 Incrementing MY_INT to : 2 Incrementing MY_INT to : 2 Incrementing MY_INT to : 2 Got Change for MY_INT : 2 Incrementing MY_INT to : 2 Incrementing MY_INT to : 2 Incrementing MY_INT to : 3 Incrementing MY_INT to : 2 Incrementing MY_INT to : 3 Incrementing MY_INT to : 3 Got Change for MY_INT : 3 Incrementing MY_INT to : 3 Incrementing MY_INT to : 3 Incrementing MY_INT to : 4 Incrementing MY_INT to : 3 Incrementing MY_INT to : 4 Incrementing MY_INT to : 4 Got Change for MY_INT : 4 Incrementing MY_INT to : 4 Incrementing MY_INT to : 4 Incrementing MY_INT to : 5 Got Change for MY_INT : 5 Incrementing MY_INT to : 4 Incrementing MY_INT to : 5 Incrementing MY_INT to : 5 Process finished with exit code 0

  

 

 

posted @ 2020-07-15 12:24  我要去巴萨  阅读(184)  评论(0编辑  收藏  举报