Lock实现原理

当多个线程需要访问某个公共资源的时候,我们知道需要通过加锁来保证资源的访问不会出问题。

java提供了两种方式来加锁,一种是关键字:synchronized,一种是concurrent包下的lock锁。

synchronized是java底层支持的,而concurrent包则是jdk实现

关于synchronized的原理可以阅读再有人问你synchronized是什么,就把这篇文章发给他。

在这里,我会用尽可能少的代码,尽可能轻松的文字,尽可能多的图来看看lock的原理。

我们以ReentrantLock为例做分析,其他原理类似。

我把这个过程比喻成一个做菜的过程,有什么菜,做法如何?

我先列出lock实现过程中的几个关键词:计数值、双向链表、CAS+自旋

我们以ReentrantLock为例做分析,其他原理类似。

实现原理

ReentrantLock() 干了啥

  public ReentrantLock() {
 
        sync = new NonfairSync();
 
    }

在lock的构造函数中,定义了一个NonFairSync

static final class NonfairSync extends Sync

NonfairSync 又是继承于Sync

abstract static class Sync extends AbstractQueuedSynchronizer

一步一步往上找,找到了

这个鬼AbstractQueuedSynchronizer(简称AQS),最后这个鬼,又是继承于AbstractOwnableSynchronizer(AOS),AOS主要是保存获取当前锁的线程对象,代码不多不再展开。最后我们可以看到几个主要类的继承关系:

 FairSync 与 NonfairSync的区别在于,是不是保证获取锁的公平性,因为默认是NonfairSync,我们以这个为例了解其背后的原理。

其他几个类代码不多,最后的主要代码都是在AQS中,我们先看看这个类的主体结构。

看看AbstractQueuedSynchronizer是个什么

再看看Node是什么?

看到这里的同学,是不是有种热泪盈眶的感觉,这尼玛,不就是双向链表么?我还记得第一次写这个数据结构的时候,发现居然还有这么神奇的一个东西。

最后我们可以发现锁的存储结构就两个东西:"双向链表" + "int类型状态"。

需要注意的是,他们的变量都被"transientvolatile修饰。

一个int值,一个双向链表是如何烹饪处理锁这道菜的呢,Doug Lea大神就是大神,

我们接下来看看,如何获取锁?

lock.lock()怎么获取锁?

public void lock() {
 
    sync.lock();
 
}

可以看到调用的是,NonfairSync.lock()

看到这里,我们基本有了一个大概的了解,还记得之前AQS中的int类型的state值

这里就是通过CAS(乐观锁)去修改state的值。lock的基本操作还是通过乐观锁来实现的。

获取锁通过CAS,那么没有获取到锁,等待获取锁是如何实现的?我们可以看一下else分支的逻辑,acquire方法:

public final void acquire(int arg) {
 
    if (!tryAcquire(arg) &&
 
        acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
 
        selfInterrupt();
 
}

这里干了三件事情:

  • tryAcquire:会尝试再次通过CAS获取一次锁。

  • addWaiter:将当前线程加入上面锁的双向链表(等待队列)中

  • acquireQueued:通过自旋,判断当前队列节点是否可以获取锁。

addWaiter() 添加当前线程到等待链表中

可以看到,通过CAS确保能够在线程安全的情况下,将当前线程加入到链表的尾部。

enq是个自旋+上述逻辑,有兴趣的可以翻翻源码。

acquireQueued()    自旋+CAS尝试获取锁

可以看到,当当前线程到头部的时候,尝试CAS更新锁状态,如果更新成功表示该等待线程获取成功。从头部移除。

每一个线程都在 自旋+CAS

最后简要概括一下,获取锁的一个流程

Java中的Lock)是用于实现线程同步的机制,确保多个线程之间对共享资源的访问是安全的。Lock接口定义了对共享资源的访问控制方法,常见的实现类有ReentrantLock和ReentrantReadWriteLock实现原理: 在Java中,实现主要依赖于底层的操作系统提供的原语,如CAS(Compare and Swap)或Mutex。CAS是一种乐观机制,通过比较并交换的方式来实现对共享资源的访问控制。Mutex是一种互斥机制,通过对共享资源进行加和解操作来实现线程之间的同步。 动态代理: 动态代理是一种在运行时生成代理对象的机制,可以在不修改原始类的情况下为其添加额外的功能。Java中的动态代理主要依赖于两个接口:InvocationHandler和Proxy。InvocationHandler定义了代理对象要执行的方法,而Proxy则负责生成代理对象。 反射: 反射是Java中一种强大的机制,它允许程序在运行时获取类的信息并操作类或对象。通过反射,可以动态地创建对象、调用方法、访问字段等。Java中的反射主要依赖于Class类和java.lang.reflect包中的其他类。 总结: Java中的机制(Lock)是实现线程同步的重要工具,它依赖于底层的操作系统提供的原语。动态代理是一种在运行时生成代理对象的机制,可以为原始类添加额外的功能。而反射是Java中一种强大的机制,允许程序在运行时获取类的信息并操作类或对象。这些特性都为Java提供了更高级别的编程能力和灵活性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值