前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >volatile关键字原理的使用介绍和底层原理解析和使用实例

volatile关键字原理的使用介绍和底层原理解析和使用实例

作者头像
青山师
发布2023-05-05 20:12:11
870
发布2023-05-05 20:12:11
举报

volatile关键字原理的使用介绍和底层原理解析和使用实例

1. volatile 关键字的作用

volatile 关键字的主要作用是保证可见性和有序性,禁止编译器优化。

  • 保证可见性:当一个变量被声明为 volatile 之后,每次读取这个变量的值都会从主内存中读取,而不是从缓存中读取,这就保证了不同线程对这个变量操作的可见性。
  • 有序性:volatile 关键字保证了不同线程对一个 volatile 变量的读写操作的有序性。
  • 禁止编译器优化:编译器会对代码进行各种优化来提高性能,但是这些优化也可能让同步代码失效。volatile 关键字告诉编译器不要对这段代码做优化,从而避免一些不正确的优化。

2. volatile 的底层原理

volatile 关键字底层原理依赖于内存屏障和缓存一致性协议。

  • 内存屏障:内存屏障会强制让读和写操作都访问主内存,从而实现可见性。volatile 写操作后会加入写屏障,volatile 读操作前会加入读屏障。
  • 缓存一致性协议:每个处理器都有自己的高速缓存,当某个处理器修改了共享变量,需要缓存一致性协议来保证其他处理器也看到修改后的值。缓存一致性协议会在读操作后和写操作前加入缓存刷新操作,保证其他处理器的缓存是最新值。

3. volatile 的使用案例

volatile 关键字常用在 DCL(Double Check Lock)单例模式中:

代码语言:javascript
复制
public class Singleton {
    private volatile static Singleton instance;

    private Singleton() {}

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

这里使用 volatile 是为了防止指令重排序,保证 instance 初始化后其他线程可以看到。

volatile 也常用在Interruptible线程中,实现线程的中断功能:

代码语言:javascript
复制
public class InterruptibleThread extends Thread {
    private volatile boolean interrupted = false;

    public void interrupt() {
        interrupted = true;
    }

    @Override
    public void run() {
        while (!interrupted) {
            // do something
        }
    }
}

这里 volatile 可以保证 interrupted 的可见性,使线程立即响应中断调用。

4. volatile 的原子性问题

volatile 关键字只能保证可见性和有序性,不能保证原子性。

对一个 volatile 变量的读写操作并不是原子的,而是可以分为读、改、写三个操作:

  • 读: 读取 volatile 变量的值
  • 改:对值进行修改
  • 写:将修改后的值写入 volatile 变量

这三个操作并不是一个原子操作,在多线程环境下可能导致数据竞争问题:

代码语言:javascript
复制
public class VolatileNoAtomicDemo {
    private volatile int counter = 0;

    public void increase() {
        counter++;  // 不是原子操作
    }
}

这里的 counter++ 实际上分为三步:

  1. 读:读取 counter 的值,假设为 x
  2. 改:x + 1
  3. 写:将 x + 1 的结果写入 counter

在多线程环境下,如果两个线程同时执行 increase 方法,很有可能达不到预期结果,这就是因为 counter++ 不是一个原子操作导致的。

5. 如何解决 volatile 的原子性问题

要解决 volatile 的原子性问题,可以使用 synchronized 或 Atomic 包中的类。

使用 synchronized:

代码语言:javascript
复制
public synchronized void increase() {
    counter++;  
}

使用 AtomicInteger:

代码语言:javascript
复制
private AtomicInteger counter = new AtomicInteger(0);

public void increase() {
    counter.getAndIncrement();
}

AtomicInteger 中的方法都是原子操作,可以解决 volatile 的原子性问题。

synchronized 会影响性能,AtomicInteger 的性能更好,所以一般优先选择 Atomic 包中的原子类。

6. volatile 的实现原理

volatile 的实现原理依赖于 JMM(Java Memory Model)中的几个概念:

  • 主内存:所有线程都可以访问的内存,存储共享变量的值。
  • 工作内存:每个线程私有的内存,用于存储线程使用的变量值。
  • 内存屏障:控制读写的顺序,用于保证特定操作的完成后才允许执行后续操作。

volatile 的实现原理是:

  1. 当一个线程修改一个volatile变量的值时,它会在变量修改后立即刷新回主内存。
  2. 当一个线程读取一个volatile变量的值时,它会直接从主内存读取,而不是从工作内存读取。
  3. 它会在读后和写前加入内存屏障,以保证指令重排不会将内存操作重排到屏障另一侧。

这样就实现了:

  • 可见性:因为每次直接读写主内存,所以每个线程都可以获得最新值。
  • 有序性:内存屏障会阻止重排,读写顺序由代码决定。
  • 禁止编译器优化:因为每次都要从主内存读写,编译器难以对其进行优化。

JMM的这几个概念配合volatile关键字的实现原理,就保证了多线程环境下volatile变量的可见性、有序性和禁止编译器优化。

7. 小结

  • volatile关键字主要保证可见性、有序性和禁止编译器优化。
  • volatile的底层原理是依赖内存屏障和缓存一致性协议实现的。
  • volatile不能保证原子性,要配合synchronized或Atomic类解决。
  • volatile的实现依赖JMM中的主内存、工作内存和内存屏障等概念。

8. volatile的最佳实践

根据volatile的特性,我们可以总结出一些最佳实践:

  1. 不要过度使用volatile volatile关键字会影响程序性能,所以不要过度使用,只在真正需要可见性和有序性保证的地方使用。
  2. 与synchronized一起使用 当需要保证原子性时,volatile关键字需要与synchronized关键字一起使用。synchronized可以保证代码块的原子性,volatile可以保证数据的可见性。
  3. 使用Atomic类代替synchronized和volatile Atomic类提供的方法都是原子操作,性能比synchronized更好,同时可以保证可见性,所以在需要保证原子性的场景可以优先选择Atomic类。
  4. 禁止把long和double类型变量声明为volatile 根据JMM规范,对64位数据类型的读写操作不一定是原子的,所以不要将long和double类型的变量声明为volatile。可以使用AtomicLong和AtomicDouble类代替。
  5. volatile不保证顺序 volatile关键字只能保证有序性,不能保证顺序。有序性是指:在一个线程内,不会由于编译器优化和处理器重新排序,使得对一个volatile变量的写操作排在读操作之前。顺序是指:两个线程访问同一个变量的顺序。所以不要依赖volatile保证线程间的顺序。
  6. volatile变量不能保护其它非volatile变量 在使用volatile变量控制住多线程变量的可见性时,不要认为它可以保护其它非volatile变量。每个变量都需要单独使用volatile或synchronized来保护。

9. 案例:使用volatile实现双重检查锁定

双重检查锁定(Double Check Locking)是一种使用同步控制并发访问的方式,可以实现延迟初始化。它通过两次对对象引用进行空检查来避免同步,从而提高性能。

但是在Java中,普通的双重检查锁定是不起作用的,原因是有指令重排的存在,可能导致另一个线程看到对象引用不是null,但是对象资源还没有完成初始化。

使用volatile关键字可以禁止指令重排,实现双重检查锁定。代码示例:

代码语言:javascript
复制
public class Singleton {
    private volatile static Singleton instance;

    private Singleton() {}

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

这里把instance变量声明为volatile,可以禁止指令重排,保证在对象完成初始化后,其他线程可以正确看到instance不为null。

这种方式是实现Singleton模式的最佳方式,它只有第一次调用getInstance方法时才会同步,这样既可以实现线程安全,又有很高的性能。

10. 案例:使用volatile实现中断机制

我们可以使用一个volatile变量作为中断标志,在循环体内检查这个变量,一次循环检查后立即重新读取变量的值,保证对变量修改的可见性,从而实现中断机制。

代码语言:javascript
复制
public class VolatileInterruptionDemo extends Thread { 
    private volatile boolean interrupted = false;

    @Override
    public void run() {
        while (!interrupted) {
            // do something
        }
        System.out.println("Interrupted!");
    }

    public void interrupt() {
        interrupted = true;
    }
}

这里的interrupted变量被声明为volatile,可以保证线程可以感知到中断信号,从循环体内退出。

这就是使用volatile实现的一种简单的中断机制,利用了volatile的可见性来保证线程可以正确读取到最新的中断标志。

11. 案例:使用AtomicInteger代替volatile

前面提到过,volatile不能保证原子性,要解决这个问题可以使用synchronized或Atomic类。这里我们通过一个例子来展示如何使用AtomicInteger代替volatile。

先看一个使用volatile的例子:

代码语言:javascript
复制
public class VolatileDemo {
    private volatile int counter = 0;

    public void increase() {
        counter++;
    }

    public int getCounter() {
        return counter;
    }
} 

这里的counter++不是一个原子操作,在多线程环境下会存在数据竞争问题。

现在使用AtomicInteger代替:

代码语言:javascript
复制
public class AtomicDemo {
    private AtomicInteger counter = new AtomicInteger(0);

    public void increase() {
        counter.getAndIncrement();
    }

    public int getCounter() {
        return counter.get();
    }
}

AtomicInteger的getAndIncrement()方法是一个CAS原理的原子操作,可以保证线程安全。

AtomicInteger使用CAS操作实现原子操作,CAS操作包含三个操作:

  1. 获取变量的当前值V
  2. 对V的值进行操作
  3. 使用CAS操作设置变量的值,这个设置值的操作需要提供变量的当前值V和新值,当变量的当前值还是V时才会设置新值,否则重新获取当前值。

CAS操作可以保证如果在多个线程同时使用一个变量时,只有一个线程可以更新变量的值,其他线程的设置值操作都会失败,这种机制可以实现原子操作。

所以,通过这个例子我们可以看出,AtomicInteger是一个很好的替代volatile的选择,它可以保证原子性也具有volatile所有特性,性能也更好,是实现原子操作的最佳选择。

12. 案例:基于volatile实现一个简单的并发容器

这里我们实现一个简单的线程安全的容器,它只包含两个方法:add()和size()。

使用volatile和synchronized实现如下:

代码语言:javascript
复制
public class VolatileContainer {
    private volatile int size = 0;
    private Object[] items = new Object[10];
    
    public void add(Object item) {
        synchronized (items) {
            items[size] = item;
            size++;
        }
    }
    
    public int size() {
        return size;
    }
}

这里使用volatile声明size变量来保证线程安全,同时使用synchronized对items数组加锁来保证添加操作的原子性。

size()方法只需要简单的读取size变量,由于它被声明为volatile,可以保证每次得到的都是最新大小值。

这是一个使用volatile和synchronized实现的简单线程安全容器,利用了volatile的可见性和synchronized的互斥锁来保证线程安全。

相比直接对整个方法加锁,这种方式的性能会更好,因为size()方法没有加锁,可以并发执行,只有在必要的add()方法进行同步,这也体现了锁的精确性原则。

13. 小结

通过这几个案例,加深了对volatile和AtomicInteger的理解,主要体会到:

  1. volatile可以保证可见性和有序性,但不能保证原子性,要用synchronized或Atomic类补充。
  2. AtomicInteger可以完全替代volatile,并且性能更好,是原子操作的最佳选择。
  3. 合理使用volatile和锁可以实现较高性能的线程安全程序。锁的使用要遵循精确性原则,不要过度使用。
  4. volatile和AtomicInteger都是JMM的重要组成部分,理解它们的实现原理有助于使用它们。

14. 案例:使用AtomicStampedReference实现ABA问题的解决

ABA问题是这样的:如果一个变量V初次读取的值是A,它的值被改成了B,后来又被改回为A,那些个依赖于V没有发生改变的线程就会产生错误的依赖。

这个问题通常发生在使用CAS操作的并发环境中,我们可以使用版本号的方式来解决这个问题,每次变量更新的时候版本号加1,那么A->B->A这个过程就会被检测出来。

AtomicStampedReference就是用过这个原理来解决ABA问题的,它包含一个值和一个版本号,我们可以这样使用:

代码语言:javascript
复制
AtomicStampedReference<Integer> atomicRef = 
    new AtomicStampedReference<>(100, 0);

// 获取当前值和版本号  
int stamp = atomicRef.getStamp();
int value = atomicRef.getReference();

// 尝试设置新值和版本号
boolean success = atomicRef.compareAndSet(value, 101, stamp, stamp + 1);
if(success) {
    // 设置成功,获取新版本号
    stamp = atomicRef.getStamp(); 
}   

这里当我们重新设置值100的时候,由于版本号已经变了,所以compareAndSet会失败,ABA问题就被解决了。

AtomicStampedReference是JUC包中用来解决ABA问题的重要工具类,实际项目中也广泛使用,它利用版本号的方式巧妙解决了这个并发编程中容易产生的问题。

另外,AtomicStampedReference的版本号使用的是int类型,所以在高并发场景下也可能存在循环的问题,这个时候可以使用时间戳方式生成版本号来避免,不过一般情况下AtomicStampedReference已经可以很好解决ABA问题。

15. 总结

OK,到这里volatile相关内容就全部介绍完了,包括:

  1. volatile的定义及作用:可见性、有序性和禁止优化。
  2. volatile的底层实现原理:JMM、缓存一致性协议和内存屏障。
  3. volatile的使用实例:双重检查锁定和中断机制等。
  4. 如何解决volatile的原子性问题:使用synchronized和Atomic类。
  5. AtomicStampedReference用法和ABA问题解决。
  6. 一些volatile的最佳实践。
  7. 使用volatile和锁实现的一个简单线程安全容器。

讲解的内容比较广泛,试着结合理论和实践的方式进行解释,希望可以对大家理解volatile和并发编程有所帮助。这也是我学久而久之总结的一些心得体会,与大家共同分享学习。如果 对volatile和JMM还有哪些不理解的地方,也欢迎留言讨论,我们共同进步!再次感谢阅读这篇博客,也希望您能够在学习和工作中很好地应用volatile关键字!

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2023-05-02,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • volatile关键字原理的使用介绍和底层原理解析和使用实例
    • 1. volatile 关键字的作用
      • 2. volatile 的底层原理
        • 3. volatile 的使用案例
          • 4. volatile 的原子性问题
            • 5. 如何解决 volatile 的原子性问题
              • 6. volatile 的实现原理
                • 7. 小结
                  • 8. volatile的最佳实践
                    • 9. 案例:使用volatile实现双重检查锁定
                      • 10. 案例:使用volatile实现中断机制
                        • 11. 案例:使用AtomicInteger代替volatile
                          • 12. 案例:基于volatile实现一个简单的并发容器
                            • 13. 小结
                              • 14. 案例:使用AtomicStampedReference实现ABA问题的解决
                                • 15. 总结
                                相关产品与服务
                                容器服务
                                腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
                                领券
                                问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档