重学Volatile

简介: 并发,是多个线程去访问同一个资源;并行,各种事情同时在做。volatile是java虚拟机提供的**轻量级**的**同步机制**三大特性,保证可见性,不保证原子性,禁止指令重排序先说下JMM(java内存模型Java Memory Model)本身是一种抽象的概念并不是真实存在的,它描述的是一组规则或规范,通过这组规范定义了程序中各个变量(包括实例字段,静态字段和构成数组对象的元素)的访问方式。

并发,是多个线程去访问同一个资源;并行,各种事情同时在做。

volatile是java虚拟机提供的轻量级同步机制

三大特性,保证可见性,不保证原子性,禁止指令重排序

先说下JMM(java内存模型Java Memory Model)本身是一种抽象的概念并不是真实存在的,它描述的是一组规则或规范,通过这组规范定义了程序中各个变量(包括实例字段,静态字段和构成数组对象的元素)的访问方式。

JMM关于同步的规定:

  • 1、线程解锁前,必须把共享变量的值刷新回主内存;
  • 2、线程加锁前,必须读取主内存的最新值到自己的工作内存
  • 3、加锁解锁是同一把锁

要求可见性,保证原子性、有序性,但是volatile只能保证两个,所以说volatile是轻量级的同步机制。

由于JVM运行程序的实体是线程,而每个线程创建时JVM都会为其创建一个工作内存(有些地方称为栈空间),工作内存是每个线程的私有数据区域,而Java内存模型中规定所有变量都存储在主内存中,主内存是共享内存区域,所有线程都可以访问,但线程对变量的操作(读取赋值等)必须在工作内存中进行,首先将变量从主内存拷贝到自己的工作内存空间,然后对变量进行操作,操作完成后再将变量写回主内存,不能直接操作主内存中的变量,各个线程中的工作内存中存储着主内存中的变量副本考本,因此不同的线程间无法访问对方的工作内存,线程间的通信(传值)必须通过主内存来完成,如图所示。
image.png

一、volatile 保证可见性,就是数据都存放在主内存中,各个线程想要修改主内存的数据时,需要把主内存的数据,复制一份到自己的工作内存中,修改数据后,把数据再写回给主内存,其他线程再从主内存中获取最新的数据值。 代码验证

class MyData{
   volatile int number=0;
    public void addNumber(){
        this.number=60;
    }
  }
 
public class voliteDemo{
    public static void main(String args[]){
        MyData myData = new MyData();
        
        new Thread(()->{
        System.out.println(Thread.currentThread().getName()+"\t come in");
        try{
        TimeUnit.SECONDS.sleep(3);  
        }catch(InterruptedException e){
            e.printStackTrace();
        }
        myData.addNumber();
        System.out.println(Thread.currentThread().getName()+"\t update number value" + myData.number);
        
        },"AAA").start();
        
        while(myData.number==0){}
        
        System.out.println(Thread.currentThread().getName()+"\t main thread  vaule "+ myData.number);
         
    }
 
}

总结,线程AAA,和主线程是两个线程。线程AAA,修改了变量的值后,如果不加volatile修饰变量,主线程是不知道变量变更的,如果加了volatile修饰变量,修改变量的值后,主线程是可以知道修改后的变量的值。添加volatile关键字修饰,可以保证可见性。

二、volatile不保证原子性,什么是原子性呢,不可分割,完整性,即某个线程正在处理某个业务的时候,中间不可以被加塞或者被分割,需要整体完整要么同时成功,要么同时失败。

class MyData{
    volatile int number = 0;
   
    public void addPlusPlus(){
        number++;
       
    }
}
 
 
class MyDataDemo{
    public static void main(String args[]){
        MyData myData = new MyData();
        for(int i =1;i<=20;i++){
              new Thread(()->{
                  for(j=1;j<=1000;j++){
                  myData.addPlusPlus();
                  }
              },String.valueof(i)).start();
        }
        while(Thread.activeCount() >2){
            Thread.yield();
        }
        
        System.out.println(Thread.currentThread().getName()+"\t number value "+myData.number);
    
    }
 
}

运行之后number的值会小于20000,说明volatile不能保证原子性。多线程访问方法,会出现丢数据的情况。说明下为什么不能保证原子性,因为多个线程去修改主内存中的变量时,都是把主内存的变量复制到自己的工作内存中,每个线程修改完变量后,再回写回主内存。当一个线程修改变量的值后,回写回主内存和其他的线程回写回主内存的值是一样的,所以被覆盖了,就会出现丢数据的情况。 解决保证原子性,可以使用synchronized,但是使用synchronized太重了,不建议使用。可以使用AtomicInterger。修改代码

class MyData{
    volatile int number =0;
    
    AtomicInteger atomicInteger = new AtomicInteger();
    
    public void addAtomic(){
        atomicInteger.getAndIncrement();
    }
 
}
 
 
public class VolatileDemo{
 
    public static void main(String args[]){
    
    MyData myData = new MyData();
    
    for(inti=0; i<20;i++){
         new Thread(()->{
                  for(j=1;j<=1000;j++){
                  myData.addAtomic();
                  }
              },String.valueof(i)).start();
    
    }
    
  while(Thread.activeCount() >2){
            Thread.yield();
        }
        
        System.out.println(Thread.currentThread().getName()+"\t number value "+myData.atomicInteger);
 
}
 
}

这个时候就可以保证原子性 为什么加了atomic就可以解决原子性,因为使用了CAS(比较再交换)。

三、volatile,禁止指令重排序。在多线程的情况下,源码写的顺序,编译器写的顺序不一定会一样。单线程的时候,确保程序最终执行结果和代码顺序执行结果的一致性。 源代码->编译器优化的重排->指令并行的重排->内存系统的重排->最终执行的指令。

int a,b,x,y = 0;
 
 
线程1           线程2
x=a;            y=b;
b=1;            a=2;
输出x=0 y=0
如果编译器对这个段代码重排优化,
线程1           线程2
b=1;           a=2;
x=a;           y=b;
输出x= 2,y=1;

这就编译器修改了执行的顺序,带来的结果不一样。然后使用volatile关键字修饰变量后,编译器就不会再进行重排序。

线程安全性获得保证:

工作内存与主内存同步数据延迟现象导致可见性的问题,可以使用synchronized和volatile关键字解决,它们都可以使一个线程修改后的变量立即对其它线程可见。

对于指令重排序导致的可见性问题,可以使用volatile关键字解决,因为volatile可以禁止指令重排序。

哪些地方用到了volatile:

单例模式,使用双重判断,出现指令重排,使用volatile禁止指令重排。

CAS的源码,JUC的包里大量使用的volatile。

目录
相关文章
|
8月前
|
存储 缓存 安全
打工人,从 JMM 透析 volatile 与 synchronized 原理
打工人,从 JMM 透析 volatile 与 synchronized 原理
85 2
|
27天前
|
缓存 编译器 C++
第十五问:volatile是什么?有什么用?
本文深入探讨了C/C++中的`volatile`关键字,解释了其防止编译器不当优化、保证多线程间可见性和确保硬件状态正确读写的作用。同时,文章也指出了使用`volatile`可能带来的性能影响,并强调了它在多线程同步中的局限性。通过具体示例,帮助读者更好地理解和应用这一强大工具。
34 0
|
3月前
|
缓存 安全 Java
深入探索研究volatile
【10月更文挑战第16天】
23 7
|
7月前
|
缓存 安全 Java
《volatile使用与学习总结:》多层面分析学习java关键字--volatile
《volatile使用与学习总结:》多层面分析学习java关键字--volatile
39 0
|
6月前
|
存储 缓存 Java
(一) 玩命死磕Java内存模型(JMM)与 Volatile关键字底层原理
文章的阐述思路为:先阐述`JVM`内存模型、硬件与`OS`(操作系统)内存区域架构、`Java`多线程原理以及`Java`内存模型`JMM`之间的关联关系后,再对`Java`内存模型进行进一步剖析,毕竟许多小伙伴很容易将`Java`内存模型(`JMM`)和`JVM`内存模型的概念相互混淆,本文的目的就是帮助各位彻底理解`JMM`内存模型。
134 0
|
Java 编译器
synchronized 和 volatile 的区别是什么?(高薪常问)
synchronized 和 volatile 的区别是什么?(高薪常问)
94 2
|
Java
关于关键字volatile的一二
关于关键字volatile的一二
88 0
|
缓存 Java 编译器
JUC并发编程学习(十七) -5分钟搞懂volatile
JUC并发编程学习(十七) -5分钟搞懂volatile
JUC并发编程学习(十七) -5分钟搞懂volatile
|
安全 Java 编译器
面试突击51:为什么单例一定要加 volatile?
面试突击51:为什么单例一定要加 volatile?
131 0
|
存储 缓存 编解码
java并发编程的艺术(2)浅谈volatile和synchronized
java并发编程的艺术(2)浅谈volatile和synchronized
158 0