`
MauerSu
  • 浏览: 519786 次
  • 性别: Icon_minigender_1
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

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

 
阅读更多
源:http://coderbee.net/index.php/concurrent/20131115/577/comment-page-1#comment-3074
评:
自旋锁(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操作),解决了多线程并发操作导致数据不一致的问题,确保其他线程可以看到锁的真实状态。
缺点

    CAS操作需要硬件的配合;
    保证各个CPU的缓存(L1、L2、L3、跨CPU Socket、主存)的数据一致性,通讯开销很大,在多处理器系统上更严重;
    没法保证公平性,不保证等待进程/线程按照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 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
            }
        }
    }

    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 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

差异:

    从代码实现来看,CLH比MCS要简单得多。
    从自旋的条件来看,CLH是自旋在其他对象的属性,MCS是在本地变量上自旋。
引用
CLH是在前驱节点的属性上自旋,而MCS是在本地属性变量上自旋。

    从链表队列来看,CLH的队列是隐式的,CLHNode并不实际持有下一个节点;MCS的队列是物理存在的。
    CLH锁释放时只需要改变自己的属性,MCS锁释放则需要改变后继节点的属性。

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

相关推荐

    自旋锁操作 spin_lock

    自旋锁的基本思想是当一个线程试图获取已被其他线程持有的锁时,它不会立即阻塞,而是不断地循环检查锁的状态,直到锁变为可用状态为止,这个过程称为“自旋”。在Linux内核中,自旋锁是实现内核级并发的重要工具。 ...

    信号量、互斥体和自旋锁的区别

    一旦一个线程获得了互斥体的锁定,其他试图获取该互斥体的线程将被阻塞,直到当前持有者释放锁为止。互斥体特别适用于线程间的同步控制,不适用于进程间通信。 #### 四、自旋锁 自旋锁(Spin Lock)是另一种常见的...

    Linux系统内核的同步机制-自旋锁

    自旋锁的设计理念在于,当一个线程尝试获取已被其他线程持有的自旋锁时,它不会进入睡眠状态,而是会不断地进行忙循环,检查锁的状态,直到锁被释放。这种行为被称为“自旋”,因此得名自旋锁。 自旋锁的特点在于其...

    信号量与自旋锁

    2. **获取自旋锁**:`spin_lock()`用于获取自旋锁,如果锁已被占用,调用者将不断地检查锁的状态,直到锁被释放。 3. **尝试获取自旋锁**:`spin_trylock()`提供了一种非阻塞的方式尝试获取自旋锁,如果锁无法立即...

    用原子自旋读写锁代替互斥锁提高多线程访问公共资源效率

    传统的互斥锁(Mutex)是一种常见的同步机制,它通过让一个线程获得锁并独占资源,其他线程则等待该线程释放锁来实现线程间的同步。然而,在高并发环境下,互斥锁可能会成为性能瓶颈,因为它可能导致线程频繁地上...

    linux 自旋锁讨论记录

    它通过维护一个计数器来表示可用资源的数量,当进程请求资源时,会尝试减小该计数器的值;当释放资源时,则增加计数值。信号量支持进程在等待资源时进入睡眠状态,因此它适合于需要长时间保持锁的场景。 #### 三、...

    并发控制之自旋锁.pdf

    当一个线程尝试获取一个已被占用的自旋锁时,它会进入一个循环,不断检查锁是否已经释放,而不是像互斥锁那样挂起线程并让出处理器。这在多处理器系统中是有意义的,因为在多核环境中,另一个持有锁的线程可能就在同...

    linux_锁_原子_自旋

    自旋锁是一种特殊的锁机制,当一个线程试图获取已被其他线程持有的锁时,它会不断地“自旋”检查锁的状态,而不会被调度出去。一旦锁被释放,自旋停止,线程立即获得锁。自旋锁适用于持有时间短且竞争激烈的场景,...

    linux下自旋锁程序源码.zip

    自旋锁的原理是,当一个线程试图获取一个已经被其他线程持有的锁时,它不会像普通互斥锁那样进入睡眠状态,而是会不断地循环检查锁的状态,直至锁变为可用状态。这种机制在处理器等待时间短且锁持有时间也很短的情况...

    zynq的linux驱动6-使用自旋锁实现竞争保护

    在Linux内核中,自旋锁是内核级的同步机制,当一个线程试图获取已被其他线程持有的锁时,它会进入“自旋”状态,不断地检查锁是否释放,而不会将控制权交还给调度器。这样做的优点是在大多数情况下,一旦锁被释放,...

    ZYNQ 7010-7020实现自旋锁测试(Linux驱动).zip

    自旋锁的特性决定了持有锁的线程如果发现资源被占用,它会不断地循环检查("自旋"),直到锁被释放。 在ZYNQ 7010-7020上实现自旋锁测试,首先我们需要理解Linux驱动模型。Linux驱动程序通常分为字符设备驱动、块...

    彻底理解Java中的各种锁.pdf

    可重入锁也称为递归锁,它允许同一个线程多次进入被该锁保护的代码块。实现可重入锁的关键在于记录当前持有锁的线程以及锁的计数器,每当线程获取锁时,计数器就会递增,释放锁时递减,直到计数器归零,锁才被其他...

    Linux内核同步机制的自旋锁原理及综合应用实例

    这意味着,当一个线程尝试获取已被持有的自旋锁时,它会不断地检查锁的状态,直到锁变为可用。这种机制在多处理器环境下特别有用,因为线程可以在等待锁的过程中保持活跃,一旦锁被释放,就能立即获得并继续执行。 ...

    DirectShow采集音频,采用环形缓冲区对象将音频数据提供出来,还提供了自旋锁的概念

    当一个线程获取到锁时,其他试图获取锁的线程将进入“自旋”状态,不断地检查锁是否释放,一旦锁被释放,自旋的线程就会立即获得锁并继续执行,这样可以减少线程上下文切换的开销。 4. **代码实现与配置**: 在...

    原子操作、信号量、读写信号量和自旋锁的API

    4. **spin_trylock(spinlock_t *lock)**:尝试获取自旋锁,如果锁被其他线程持有,则立即返回失败。 **示例**: 在实时系统中,当需要保护某个短暂的关键代码段时,可以使用自旋锁。这样可以确保即使在高负载下,...

    Linux内核同步机制-自旋锁

    自旋锁的基本思想是当一个线程试图获取已被其他线程持有的锁时,它会不断地检查锁的状态,即“自旋”等待,直到锁被释放为止。由于自旋锁不会导致调用者进入睡眠状态,因此它主要适用于那些持有锁的时间非常短的操作...

    spinlocks:各种自旋锁实现和基准测试

    在计算机系统中,当一个线程尝试获取已经被其他线程持有的锁时,自旋锁会使得该线程“自旋”等待,而不是被挂起。自旋锁的主要目标是减少线程上下文切换的开销,提高系统的并行性能。 本文将深入探讨自旋锁的实现...

    mutexes-variables-semaphores.rar_linux 线程锁_mutexes_锁和信号量

    自旋锁是一种特殊的锁机制,当锁被占用时,尝试获取锁的线程不会进入睡眠状态,而是不断地循环检查锁是否可用,即“自旋”。这种机制在处理短时锁时效率较高,因为线程无需从内核态切换到用户态。Linux内核提供了...

    高并发多线程面试专题及答案(上).pdf

    - 自旋锁(Spin Lock):当一个线程尝试获取已经被其他线程持有的锁时,它会进行短时间的循环等待,而不是立即进入阻塞状态。 - 偏向锁(Biased Locking):偏向锁通过在Mark Word中记录持有偏向锁的线程ID来避免...

    Linux应用程序之线程通信

    自旋锁是另一种同步原语,当锁被占用时,尝试获取锁的线程会不断地检查锁的状态,而不是进入睡眠状态。在Linux中,可以使用`pthread_spinlock_t`类型和`pthread_spin_lock()`、`pthread_spin_unlock()`等函数来实现...

Global site tag (gtag.js) - Google Analytics