zoukankan      html  css  js  c++  java
  • Volatile、Synchronization 与 ReentrantLock

    Volatile可见性

    比如现在我们有这样一段代码:线程等待另一个线程将数据装载完就输出success,可是最后程序一直卡在while循环里没有往下执行。

    public class VolatileDemo {
        private static boolean flag = false;
        //private static volatile boolean flag = false;
    
        public static void main(String[] args) throws Exception{
            new Thread(()->{
                System.out.println("等待装载数据。。。。");
                while(!flag){
                }
                System.out.println("====== SUCCESS =====");
            }).start();
            Thread.sleep(2000);
            new Thread(()->{
                System.out.println("开始装载");
                flag = true;
                System.out.println("装载完毕");
            }).start();
        }
    }
    /* 控制台输出
            等待装载数据。。。。
            开始装载
            装载完毕
     */

    造成这个问题出现的原因是jmm原子操作造成的。jmm内存模型就是java内存模型、准确的说是java线程内存模型。它和cpu缓存模型类似、是基于cpu缓存模型来建立的。
    jmm一共有8种原子操作:
      read(读取):从主存读取数据
      load(载入):将内存数据读到工作内存
      use (使用):取出工作内存中的数据来计算
      assign(赋值):将计算好的值重新赋予到工作内存中
      store(存储):将工作内存数据写入主存
      write(写入):将store过去的变量值赋值给主内存中的变量
      lock(锁定):将主内存变量加锁,标识为线程独占状态
      unlock(解锁):将主存变量解锁,解锁后其他线程可以锁定该变量

      

    可以看到线程1已经把变量副本加载到工作内存了,而线程2将计算后的值存到主存之后,却没有办法告诉线程1,所以就出现了线程安全问题。其实cpu与主存交互会经过"总线"这么一个概念,cpu为了解决这种数据不一致问题有两种方案:
    总线加锁(性能太低)
      早期cpu是对总线加锁,lock住这个数据,这样其它线程就没法对它读或写,直到这个线程用完这个数据 unlock之后才能被其他线程操作。也就是说从read开始后直到write结束才释放锁。
    MESI缓存一致性协议
      多个线程将同一个数据读取到各自的缓存区后,某个cpu修改了缓存的数据之后,会立马同步给主存,这都是汇编语言实现的。其他cpu通过总线嗅探机制(可以理解为监听)可以感知到数据的变化从而将自己缓存里的数据失效,从而去读取主存的值。所以mesi协议是从store开始加锁,锁的粒度更小,时间更短。实际上volatile就是这么实现可见性的。同时由于这中间过程中有store和write几步操作、还要让其他cpu缓存的数据置空都是要耗时的,可能这个过程中数据被别人改了,所以它是非原子操作的。

    volatile与指令重排

    指令重排
      指定重排只会发生在多线程情况下,单线程是不会出现指定重排的。所谓的指令重排就是JVM在编译Java代码的时候,或者CPU在执行JVM字节码的时候,对现有的指令顺序进行排序优化。但不会对有依赖关系的做重排序。比如:
      int a = 1;
      int b = 2;
      int c = a*c;
      a 和 b 没有任何关系,所以它们的顺序无所谓,但是 c 依赖于a、b。只能存在于a、b后面,不然就乱套了。
    在一个变量被volatile修饰后会被禁止指令重排,JVM会为我们做两件事:
      1.在每个volatile写操作前插入StoreStore屏障,在写操作后插入StoreLoad屏障。
      2.在每个volatile读操作前插入LoadLoad屏障,在读操作后插入LoadStore屏障。

    Synchronization原子性

      synchronized (a){} 锁住的就是()里面的对象,多个线程对同一个对象操作时,就会形成互斥效果,如果是操作两个不同的对象,那么就不会受synchronized影响

    public class SynchronizedDemo {
        public static void main(String[] args) {
            SynchronizedDemo s = new SynchronizedDemo();
            Integer a = 1;
            Integer b = 2;
    
            new Thread(()->{
                s.sync(a);
            }).start();
            new Thread(()->{
                s.sync(b);
                //s.sync(a);
            }).start();
        }
    
        public void sync(Integer a){
            synchronized (a){
                System.out.println("线程:"+Thread.currentThread().getName()+" 获取到变量"+a);
                try {
                    Thread.sleep(8000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    在jdk中,Synchronization同步是基于Monitor对象实现的,它里面主要有两个指令:
      monitorenter: 插入到同步代码块的开始位置
      monitorexit: 插入到同步代码块结束的位置
    它们对应着JMM模型8大原子操作的lock与unlock,lock获取锁后把对象加载到工作内存,数据操作完之后重新赋值到主内存,最后unlock解锁。JVM需要保证每一个monitorenter都有一个monitorexit与之对应。为了保证在方法异常时,monitorenter和monitorexit指令也能正常配对执行,编译器会自动产生一个异常处理器,它的目的就是用来执行 异常的monitorexit指令。Monitor(监视器锁)是依赖操作系统的Mutex Lock(互斥锁)实现的,需要向内核申请资源,此时cpu将由用户态转换为内核态,它是一个低性能重量级锁。

    jdk1.6对Synchronization的优化

      jdk1.6之后就对这个synchronized锁进行了各种优化,如适应性自旋锁、轻量级锁和偏向锁,并默认开启偏向锁。从 无锁—>偏向锁—>轻量级锁—>重量级锁 ,锁升级的这个过程是不可逆的。被加锁的对象 jvm中为它定义了一种对应的数据结构,通过判断数据结构的对象头就知道目前是什么锁状态。例如通过倒数第三个bit的值 0/1 就知道目前是无锁还是偏向锁了。

    三种锁的区别

      偏向锁:仅有一个线程进入临界区(主要用于不存在锁竞争,而是一个线程多次获得锁时,为的使线程获取锁使用最小的代价(因为只需要修改获取锁的线程id就好了))
      轻量级锁:多个线程交替进入临界区(当其他线程尝试竞争偏向锁时,会升级为轻量锁)
      重量级锁:多个线程同时进入临界区

    锁的升级过程

    1. 无锁:此时还没有线程获取所得资源

      

     2. 获取偏向锁:第一个线程获取到锁就会将前面的23个bit位修改为自己线程的id,将无锁升级为偏向锁。

       

     3. 升级轻量锁:此时另一个线程尝试获取锁,发现锁里的线程id并不是自己的,就会释放锁,将对象头重的Mark Word替换为指向锁记录的指针,将其升级为轻量锁。

      

    4. 若刚才将对象头重的Mark Word替换为指向锁记录的指针失败,则会自旋(循环等待)来获取锁,此时若有另一个线程同时竞争,锁会升级为重量级锁。

       

    ReentrantLock

      ReentrantLock和Synchronization一样是并发编程的核心,Synchronization是sun公司开发,而ReentrantLock是一个叫Doug Lea的人写出来的。它控制锁的状态是通过AQS(队列同步器)来实现的,主要用到了2点技术点。

    1. volatile关键字
      在AQS中定义一个volatile修饰的int变量state,有线程获取到锁之后state就加一,其他线程发现锁被占用之后就会进入等待队列。线程释放锁之后state就会减一,然后唤醒队列中的其他线程。
    2. CAS(比较替换算法)
      我们知道volatile不是线程安全的,那么如何保证只有一个线程对state在操作呢?其实就用到了CAS算法,它是一个无锁算法是乐观锁的体现。CAS需要有3个操作数:内存地址V,旧的预期值A,即将要更新的目标值B。只有A==V的时候才把V的值修改成B,否则不做任何操作。源码调用了Unsafe类的原子方法,都是被native修饰的,整个比较并替换的操作是一个原子操作。

    ReentrantLock和Synchronization比较

      ReentrantLock和synchronized在低并发的时候性能差距不大,高并发时ReentrantLock性能要稍微高一些。虽然sync做了优化但是在竞争激烈的时候还是会从偏向锁升级为重量级锁,是用户态切换到内核态的一个过程 比较消耗资源,lock有利用CAS自旋操作来实现锁则会稍微好一点。

  • 相关阅读:
    js命名空间笔记
    css3兼容性问题归纳
    flexbox-CSS3弹性盒模型flexbox完整版教程
    JavaScript 预解析
    消除页面上的链接虚线框
    图片压缩之 PNG
    常见的前端优化技巧有哪些
    for-of循环和for-in循环的区别
    函数式编程初探
    js中同步与异步处理方法
  • 原文地址:https://www.cnblogs.com/wlwl/p/11920689.html
Copyright © 2011-2022 走看看