zoukankan      html  css  js  c++  java
  • 自旋锁、排队自旋锁、MCS锁、CLH锁

    1、实现自旋锁

    通过一个AtomicReference<Thread>类型成员变量owner,就可以实现一个自旋锁,owner属性持有当前拥有锁的线程引用,如果该引用为null,表示锁未被用,不为null则被占用。通过AtomicReference对象compareAndSet方法解决了多线程并发操作导致数据不一致问题,确保其它线程可以看到锁的真实状态。

    2、实现公平自旋锁

    通过上面简单CAS操作无法保证公平性,不能保证等待线程按照FIFO顺序获得锁。可以通过类似排队叫号方案实现公平锁:锁对象拥有一个服务号,表示正在服务的线程,还有一个排队号,每个线程获取锁前先拿一个排队号,然后不断轮询当前的服务号是否是自己的排队号,若是就拥有锁,否则继续轮询。释放锁时将服务号加1,从而使下一个线程退出自旋。但这个方法多个线程都在读写同一个变量服务号,每次操作都会导致多个处理器缓存之间同步,增加系统总线和内存流量,降低系统整体性能。

    3、MCS锁和CLH锁

    因此又产生了MCS锁和CLH锁。MCS锁是基于链表的可扩展、高性能、公平自旋锁,申请线程只在本地变量上自旋,直接前驱负责通知其自旋结束,从而减少了处理器缓存同步次数,降低了系统总开销。

    CLH锁是基于隐形链表的可扩展、高性能、公平的自旋锁,申请线程只在本地变量上自旋,它不断轮询前驱的状态,发现前驱释放了锁就结束自旋。CLH锁代码实现上比MCS要简单的多,它在释放锁时只改变自己的属性,而MCS锁释放锁时需要改变后继节点的属性。

    自旋锁(SPIN LOCK)

    自旋锁是指当一个线程尝试获取某个锁时,如果该锁已被其他线程占用,就一直循环检测锁是否被释放,而不是进入线程挂起或睡眠状态。

    自旋锁适用于锁保护的临界区很小的情况,临界区很小的话,锁占用的时间就很短。

    简单的实现

    import java.util.concurrent.atomic.AtomicReference;
    
    public class SpinLock {
       private AtomicReference<Thread> owner = new AtomicReference<Thread>();
    
       public void lock() {
           Thread currentThread = Thread.currentThread();
    
                  // 如果锁未被占用,则设置当前线程为锁的拥有者
           while (!owner.compareAndSet(null, currentThread)) {
           }
       }
    
       public void unlock() {
           Thread currentThread = Thread.currentThread();
    
                  // 只有锁的拥有者才能释放锁
           owner.compareAndSet(currentThread, null);
       }
    }
    

    SimpleSpinLock里有一个owner属性持有锁当前拥有者的线程的引用,如果该引用为null,则表示锁未被占用,不为null则被占用。

    这里用AtomicReference是为了使用它的原子性的compareAndSet方法(CAS操作),解决了多线程并发操作导致数据不一致的问题,确保其他线程可以看到锁的真实状态。

    缺点

    1. CAS操作需要硬件的配合;
    2. 保证各个CPU的缓存(L1、L2、L3、跨CPU Socket、主存)的数据一致性,通讯开销很大,在多处理器系统上更严重;
    3. 没法保证公平性,不保证等待进程/线程按照FIFO顺序获得锁。

    TICKET LOCK

    Ticket Lock 是为了解决上面的公平性问题,类似于现实中银行柜台的排队叫号:锁拥有一个服务号,表示正在服务的线程,还有一个排队号;每个线程尝试获取锁之前先拿一个排队号,然后不断轮询锁的当前服务号是否是自己的排队号,如果是,则表示自己拥有了锁,不是则继续轮询。

    当线程释放锁时,将服务号加1,这样下一个线程看到这个变化,就退出自旋。

    简单的实现

    import java.util.concurrent.atomic.AtomicInteger;
    
    public class TicketLock {
       private AtomicInteger serviceNum = new AtomicInteger(); // 服务号
       private AtomicInteger ticketNum = new AtomicInteger(); // 排队号
    
       public int lock() {
             // 首先原子性地获得一个排队号
             int myTicketNum = ticketNum.getAndIncrement();
    
                  // 只要当前服务号不是自己的就不断轮询
           while (serviceNum.get() != myTicketNum) {
           }
    
           return myTicketNum;
        }
    
        public void unlock(int myTicket) {
            // 只有当前线程拥有者才能释放锁
            int next = myTicket + 1;
            serviceNum.compareAndSet(myTicket, next);
        }
    }
    

    缺点

    Ticket Lock 虽然解决了公平性的问题,但是多处理器系统上,每个进程/线程占用的处理器都在读写同一个变量serviceNum ,每次读写操作都必须在多个处理器缓存之间进行缓存同步,这会导致繁重的系统总线和内存的流量,大大降低系统整体的性能。

    下面介绍的CLH锁和MCS锁都是为了解决这个问题的。

    MCS 来自于其发明人名字的首字母: John Mellor-Crummey和Michael Scott。

    CLH的发明人是:Craig,Landin and Hagersten。

    MCS锁

    MCS Spinlock 是一种基于链表的可扩展、高性能、公平的自旋锁,申请线程只在本地变量上自旋,直接前驱负责通知其结束自旋,从而极大地减少了不必要的处理器缓存同步的次数,降低了总线和内存的开销。

    import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
    
    public class MCSLock {
        public static class MCSNode {
            volatile MCSNode next;
            volatile boolean isBlock = true; // 默认是在等待锁
        }
    
        volatile MCSNode queue;// 指向最后一个申请锁的MCSNode
        private static final AtomicReferenceFieldUpdater<mcslock, mcsnode=""> UPDATER = AtomicReferenceFieldUpdater
                .newUpdater(MCSLock.class, MCSNode.class, "queue");
    
        public void lock(MCSNode currentThread) {
            MCSNode predecessor = UPDATER.getAndSet(this, currentThread);// step 1
            if (predecessor != null) {
                predecessor.next = currentThread;// step 2
    
                while (currentThread.isBlock) {// step 3
                }
            }else { // 只有一个线程在使用锁,没有前驱来通知它,所以得自己标记自己为非阻塞
                   currentThread. isBlock = false;
              }
        }
    
        public void unlock(MCSNode currentThread) {
            if (currentThread.isBlock) {// 锁拥有者进行释放锁才有意义
                return;
            }
    
            if (currentThread.next == null) {// 检查是否有人排在自己后面
                if (UPDATER.compareAndSet(this, currentThread, null)) {// step 4
                    // compareAndSet返回true表示确实没有人排在自己后面
                    return;
                } else {
                    // 突然有人排在自己后面了,可能还不知道是谁,下面是等待后续者
                    // 这里之所以要忙等是因为:step 1执行完后,step 2可能还没执行完
                    while (currentThread.next == null) { // step 5
                    }
                }
            }
    
            currentThread.next.isBlock = false;
            currentThread.next = null;// for GC
        }
    }
    

    CLH锁

    CLH锁也是一种基于链表的可扩展、高性能、公平的自旋锁,申请线程只在本地变量上自旋,它不断轮询前驱的状态,如果发现前驱释放了锁就结束自旋。

    import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
    
    public class CLHLock {
        public static class CLHNode {
            private volatile boolean isLocked = true; // 默认是在等待锁
        }
    
        @SuppressWarnings("unused" )
        private volatile CLHNode tail ;
        private static final AtomicReferenceFieldUpdater<CLHLock, CLHNode> UPDATER = AtomicReferenceFieldUpdater
                      . newUpdater(CLHLock.class, CLHNode .class , "tail" );
    
        public void lock(CLHNode currentThread) {
            CLHNode preNode = UPDATER.getAndSet( this, currentThread);
            if(preNode != null) {//已有线程占用了锁,进入自旋
                while(preNode.isLocked ) {
                }
            }
        }
    
        public void unlock(CLHNode currentThread) {
            // 如果队列里只有当前线程,则释放对当前线程的引用(for GC)。
            if (!UPDATER .compareAndSet(this, currentThread, null)) {
                // 还有后续线程
                currentThread. isLocked = false ;// 改变状态,让后续线程结束自旋
            }
        }
    }
    

    CLH锁 与 MCS锁 的比较

    下图是CLH锁和MCS锁队列图示:
    CLH-MCS-SpinLock

    差异:

    1. 从代码实现来看,CLH比MCS要简单得多。
    2. 从自旋的条件来看,CLH是在前驱节点的属性上自旋,而MCS是在本地属性变量上自旋。
    3. 从链表队列来看,CLH的队列是隐式的,CLHNode并不实际持有下一个节点;MCS的队列是物理存在的。
    4. CLH锁释放时只需要改变自己的属性,MCS锁释放则需要改变后继节点的属性。

    注意:这里实现的锁都是独占的,且不能重入的。

    https://coderbee.net/index.php/concurrent/20131115/577

  • 相关阅读:
    SQL 2008R2问题:用户、组或角色'XXX'在当前数据库中已存在?
    修改sqlserver 2008 R2 实例名称
    keepalived vip做网关
    Django(HttpResponse、render,、redirect)的用法
    Linux脚本中$#、$0、$1、$@、$*、$$、$?
    linux定时删除历史日志文件实现方式--shell脚本
    Long转换为date
    java.lang.ClassNotFoundException: org.springframework.web.util.IntrospectorCleanupListener
    2016年新年伊始
    linux下环境搭建
  • 原文地址:https://www.cnblogs.com/doit8791/p/9098188.html
Copyright © 2011-2022 走看看