volatile关键字详解

1 volatile作用

volatile关键字的主要作用是使变量在多个线程间可见,方式是强制性从公共堆栈中进行取值。

先看个例子:

public class RunThread extends Thread{
    private boolean isRunning = true;

    public boolean isRunning() {
        return isRunning;
    }

    public void setRunning(boolean running) {
        isRunning = running;
    }

    @Override
    public void run() {
        System.out.println("进入 run 了");
        while (isRunning == true){

        }
        System.out.println("线程被停止了");
    }
}
public class TestMain {
    public static void main(String[] args) {
        try {
            RunThread thread = new RunThread();
            thread.start();
            Thread.sleep(1000);
            thread.setRunning(false);
            System.out.println("已经赋值为false");
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }
}

运行结果如下:
在这里插入图片描述

程序会一直运行下去,造成死循环。因为在启动RunThread.java 线程时,变量private boolean isRunning = true;存在于公共堆栈及线程的私有堆栈中。线程一直在私有堆栈中取得isRunning的值是true。而代码thread.setRunning(false) 虽然被执行,更新的确实公共堆栈中的isRunning变量值false,所以一直就是死循环状态。内存结构如下图所示:
线程的私有堆栈
​ 线程的私有堆栈

这个问题是私有堆栈中的值和公共堆栈中的值不同不造成的。解决这样的问题就要使用volatile关键字了,它主要的作用就是当线程访问isRunning这个变量时,强制性从公共堆栈中进行取值。

更改后RunThread.java 代码如下:

public class RunThread extends Thread{
    volatile private boolean isRunning = true;

    public boolean isRunning() {
        return isRunning;
    }

    public void setRunning(boolean running) {
        isRunning = running;
    }

    @Override
    public void run() {
        System.out.println("进入 run 了");
        while (isRunning == true){

        }
        System.out.println("线程被停止了");
    }
}
public class TestMain {
    public static void main(String[] args) {
        try {
            RunThread thread = new RunThread();
            thread.start();
            Thread.sleep(1000);
            thread.setRunning(false);
            System.out.println("已经赋值为false");
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }
}

运行结果如下:
在这里插入图片描述

通过使用volatile关键字,强制的从公共内存中读取变量的值,内存结构如下图所示

在这里插入图片描述
​ 读取公共内存

使用volatile关键字增加了实例变量在多个线程之间的可见性。但volatile关键字最致命的缺点是不支持原子性。

原子性(Atomicity):指事务的不可分割性,一个事物的所有操作要么不间断地全部被执行,要么一个也没有执行。

2 volatile非原子的特性

示例如下:

public class MyThread extends Thread{
    volatile public static int count;
    private static void addCount(){
        for (int i =0; i < 1000; i++){
            count++;
        }
        System.out.println("count =  " + count);
    }

    @Override
    public void run() {
        addCount();
    }
}
public class TestRun {
    public static void main(String[] args) {
        MyThread[] arr = new MyThread[100];
        for (int i = 0; i < 100; i++){
            arr[i] = new MyThread();
        }

        for (int j = 0; j < 100; j++){
            arr[j].start();
        }
    }
}

运行结果如下:

在这里插入图片描述

更改MyThread.java 文件代码如下:

public class MyThread extends Thread{
    volatile public static int count;

    //注意一定要添加static关键字
    //这样synchronized与static锁的内容就是MyThread.class 类了,也就达到同步的效果了。
    synchronized private static void addCount(){
        for (int i =0; i < 1000; i++){
            count++;
        }
        System.out.println("count =  " + count);
    }

    @Override
    public void run() {
        addCount();
    }
}
public class TestRun {
    public static void main(String[] args) {
        MyThread[] arr = new MyThread[100];
        for (int i = 0; i < 100; i++){
            arr[i] = new MyThread();
        }

        for (int j = 0; j < 100; j++){
            arr[j].start();
        }
    }
}

运行结果如下:
在这里插入图片描述

在本示例中,如果在方法private static void addCount()前加入synchronized同步关键字,也就没有必要再使用volatile关键字来声明count变量了。

关键字volatile主要使用的场合是在多个线程中可以感知实例变量被更改了,并且可以获得最新的 值使用,也就是用多线程读取共享变量时可以获得最新值使用。

关键字volatile提示线程每次从共享内存中读取变量,而不是从私有内存中读取,这样就保证了同步数据的可见性。但在这里需要注意的是:如果修改实例变量中的数据,比如i++,也就是i= i+1,则这样的操作其实并不是一个原子操作,也就是非线程安全的。表达式i++的操作步骤分解如下:

1 从内存中取出i的值

2 计算i的值

3 将i的值写到内存中

假如在第二步计算值的时候,另外一个线程也修改i的值,那么这个时候就会出现脏数据。解决的办法其实就是使用synchronized关键字。所以说volatile本身并不处理数据的原子性,而是强制对数据的读写及时影响到主内存的。

用图演示关键字volatile出现非线程安全的原因,变量在内存中工作的过程如下图所示。
在这里插入图片描述

可以得出以下结论:

1 read 和 load 阶段:从主存复制变量到当前线程工作内存;

2 use 和 assign 阶段:执行代码,改变共享变量值。

3 store 和 write 阶段:用工作内存数据刷新主存对应变量的值。

在多线程环境中,use 和 assign 是多次出现的,但这一操作并不是原子性,也就是在read 和 load 之后,如果主内存count变量发生修改之后,线程工作内存中的值由于已经加载,不会产生对应的变化 ,也就是私有内存和公共内存中的变量不同步,所以计算出来的结果会和预期不一样,也就出现了非线程安全问题。

对于用volatile修饰的变量,jvm虚拟机只是保证从主内存加载到线程工作内存的值是最新的。例如线程1和线程2在进行read和load的操作中,发现主内存中count的值都是5,那么都会加载这个最新的值。也就是说,volatile关键字解决的是变量读时的可见性问题,但无法保证原子性,对于多个线程访问同一个实例变量还是需要加锁同步。

3 原子类也并不完全安全

示例如下

public class MyService {
    public static AtomicLong al = new AtomicLong();

    public void addNum(){
        System.out.println(Thread.currentThread().getName() + " 加了100 之后的值是 : " + al.addAndGet(100));
        al.addAndGet(1);
    }
}
public class MyThread extends Thread{
    private MyService myService;

    public MyThread(MyService myService) {
        this.myService = myService;
    }

    @Override
    public void run() {
        super.run();
        myService.addNum();
    }
}
public class TestRun {
    public static void main(String[] args) {
        try {
            MyService service = new MyService();
            MyThread[] array = new MyThread[5];
            for (int i = 0; i< array.length; i++){
                array[i] = new MyThread(service);
            }
            for (int j = 0; j <array.length; j++){
                array[j].start();
            }
            Thread.sleep(1000);
            System.out.println(service.al.get());
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }
}

运行结果如下:
在这里插入图片描述

打印顺序出错了,应该每加1次100再加一次1.出现这样的情况是因为addAndGet()方法是原子的,但方法和方法之间的调用却不是原子的。

更改后的代码如下:

public class MyService {
    public static AtomicLong al = new AtomicLong();

    synchronized public void addNum(){
        System.out.println(Thread.currentThread().getName() + " 加了100 之后的值是 : " + al.addAndGet(100));
        al.addAndGet(1);
    }
}
public class MyThread extends Thread{
    private MyService myService;

    public MyThread(MyService myService) {
        this.myService = myService;
    }

    @Override
    public void run() {
        super.run();
        myService.addNum();
    }
}
public class TestRun {
    public static void main(String[] args) {
        try {
            MyService service = new MyService();
            MyThread[] array = new MyThread[5];
            for (int i = 0; i< array.length; i++){
                array[i] = new MyThread(service);
            }
            for (int j = 0; j <array.length; j++){
                array[j].start();
            }
            Thread.sleep(1000);
            System.out.println(service.al.get());
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }
}

运行结果如下:
在这里插入图片描述

从运行结果可以看到,是每次加100再加1,这就是我们想要得到的过程,结果是505的同时还保证在过程中累加的顺序也是正确的。

4 原子类和volatile区别

volatile 和原子类的使用场景是不一样的,如果我们有一个可见性问题,那么可以使用volatile关键字,但如果我们的问题是一个组合操作,需要用同步来解决原子性问题的话,那么可以使用原子变量。

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: volatileJava中的一个关键字,用于修饰变量。它的作用是告诉编译器,该变量可能会被多个线程同时访问,因此需要特殊处理,以保证线程安全。 具体来说,volatile关键字有以下几个特点: 1. 可见性:当一个线程修改了volatile变量的值,其他线程能够立即看到这个修改。 2. 有序性:volatile变量的读写操作会按照程序的顺序执行,不会被重排序。 3. 不保证原子性:虽然volatile变量能够保证可见性和有序性,但是它并不能保证多个线程同时修改变量时的原子性。 因此,如果需要保证原子性,需要使用synchronized关键字或者Lock接口来进行同步。 总之,volatile关键字Java中用于保证多线程访问变量的安全性的一种机制,它能够保证可见性和有序性,但是不能保证原子性。 ### 回答2: Java中的volatile关键字是一种轻量级的同步机制,用于确保多个线程之间的可见性和有序性。它可以用于修饰变量、类和方法。 1. 修饰变量:当一个变量被volatile修饰时,它会被立即写入到主内存中,并且每次读取变量时都会从主内存中重新获取最新的值。这样可以保证多个线程操作同一个变量时的可见性和一致性。 2. 修饰类:当一个类被volatile修饰时,它的实例变量就会被同步,而且每个线程都会获取最新的变量值。这样可以保证多线程操作同一对象时的可见性和一致性。 3. 修饰方法:当一个方法被volatile修饰时,它的调用会插入内存栅栏(memory barrier)指令,这可以保证方法调用前的修改操作都已经被写入主内存中,而方法调用后的读取操作也会重新从主内存中读取最新值。这样可以确保多线程之间的调用顺序和结果可见性。 需要注意的是,volatile并不能完全取代synchronized关键字,它只适用于并发度不高的场景,适用于只写入不读取的场景,不能保证复合操作的原子性。 总之,volatile关键字Java中具有广泛的应用,可以保证多线程之间的数据同步和可见性,但也需要谨慎使用,以免造成数据不一致和性能问题。 ### 回答3: Java中的volatile关键字意味着该变量在多个线程之间共享,并且每次访问该变量时都是最新的值。简单来说,volatile保证了线程之间的可见性和有序性。下面我们详细解释一下volatile的用法和作用。 1. 线程之间的可见性 volatile关键字保证了对该变量的读写操作对所有线程都是可见的。在没有用volatile关键字修饰变量的情况下,如果多个线程并发访问该变量,每个线程都会从自己的线程缓存中读取该变量的值,而不是直接从主存中读取。如果一个线程修改了该变量的值,但是其他线程不知道,那么可能导致其他线程获取到的数据不是最新的,从而引发一系列问题。而用了volatile关键字修饰该变量后,每次修改操作都会立即刷新到主存中,其他线程的缓存中的变量值也会被更新,从而保证了线程之间的可见性。 2. 线程之间的有序性 volatile关键字也保证了线程之间的有序性。多个线程并发访问同一个volatile变量时,JVM会保证每个线程按照程序指定的顺序执行操作。例如,在一个变量被volatile修饰的情况下,多个线程同时对该变量进行读写操作,JVM会保证先执行写操作的线程能够在后续的读操作中获取到最新的变量值。这么做的好处是,可以避免出现线程间操作顺序的乱序问题,从而保证了程序的正确性。 需要注意的是,并不是所有的变量都需要用volatile关键字修饰。只有在多个线程之间共享变量并且对变量的读写操作之间存在依赖关系的情况下,才需要使用volatile关键字。此外,volatile关键字不能保证原子性,如果需要保证操作的原子性,需要使用synchronized或者Lock等其他并发工具。 总之,volatile关键字Java中非常重要的关键字之一,它可以在多个线程之间保证可见性和有序性,从而保证了程序的正确性。在开发过程中,我们应该根据具体情况来选择是否使用volatile关键字,以及如何使用它。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值