Java中的ReentrantLock实现原理及代码演示

 更新时间:2024年01月13日 09:39:17   作者:java架构师-太阳  
这篇文章主要介绍了Java中的ReentrantLock实现原理及代码演示,非公平锁 如果已经进入队列,链表里面的线程是先进先出,如果已经释放了锁,在抢占锁时,链表里面的头结点和还没有入队列的线程抢锁,需要的朋友可以参考下

介绍

互斥锁 实现Lock接口 并且最好在 finally 块中释放

公平锁 非公平锁 如果已经进入队列,链表里面的线程是先进先出,如果已经释放了锁,在抢占锁时,链表里面的头结点和还没有入队列的线程抢锁

使用代码

public class Test {
    public static void main(String[] args) {
       Task task = new Task();
       for(int i = 0; i < 10; i++) {
          new Thread(task).start();
       }
    }
}
class Task implements Runnable {
    Lock lock = new ReentrantLock();
    @Override
    public void run() {
        lock.lock();
        try {
            Thread.sleep(1000);
            System.out.println("业务代码");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
}

实现原理

采用AQS+CAS+LockSupport用来阻塞和唤醒线程)

ReentrantLock有三类内部类,实现都在其内部类Sync中,默认是使用非公平锁NonFairSync。

非公平锁可提高效率,在可重入锁时可以减少线程切换开销。可以通过构造方法切换公平和非公平

Sync父类AbstractQueuedSynchronize(AQS) 此处的锁具备synchronized功能,即可以阻塞一个线程。

为了实现一把具有阻塞或唤醒功能的锁,需要几个核心要素

(1) state变量,标记锁状态。至少有两个值0/1。对state的操作,使用CAS保证线程安全

  • AbstractQueuedSynchronizer类里有变量private volatile int state 记录锁状态
  • state=0,没有线程持有锁,exclusiveOwnerThread=null
  • state=1,有一个线程持有锁,exclusiveOwnerThread=该线程
  • state > 1,说明该线程重入了该锁,等于几就重入了几次

(2) 需要记录当前是哪个线程持有锁

  • AbstractOwnableSynchronizer里面有变量private transient Thread exclusiveOwnerThread;记录锁持有的线程

(3) 需要底层支持对一个线程进行阻塞或唤醒操作

  • public native void unpark(Object thread); // 唤醒某一个线程
  • public native void park(boolean isAbsolute, long time); // 阻塞某一个线程 实现一个线程对另外一个线程的精准唤醒
  • 一般使用LockSupport的工具类,对上面两个方法进行了封装
  • 当前线程中调用park()就会被阻塞 另一个线程调用unpark(Thread t)传入被阻塞线程就可唤醒阻塞在park()地方的线程

(4) 需要有一个队列维护所有阻塞的线程。这个队列也必须是线程安全的无锁队列,也需要使用CAS对队列进行增加或删除

public abstract class AbstractQueuedSynchronizer {
  // 双向链表
  static final class Node {
  volatile Thread thread;  // 每个Node对应一个被阻塞的线程
  volatile Node prev;      // 前一个
  volatile Node next;      // 后一个
 }
private transient volatile Node head;   // 头
private transient volatile Node tail;   // 尾
}

阻塞队列是整个AQS核心中的核心。如下图所示,head指向双向链表头部,tail指向双向链表尾部。入队就是把新的Node加到tail后面,然后对tail进行CAS操作;

出队就是对head进行CAS操作,把head向后移一个位置

在这里插入图片描述

初始的时候,head=tail=NULL;然后,在往队列中加入阻塞的线程时,会新建一个空的Node,让head和tail都指向这个空Node;之后,在后面加入被阻塞的线程对象。

所以,当head=tail的时候,说明队列为空。

到此这篇关于Java中的ReentrantLock实现原理及代码演示的文章就介绍到这了,更多相关ReentrantLock实现原理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 图片叠加效果Java代码实现

    图片叠加效果Java代码实现

    这篇文章主要为大家详细介绍了图片叠加效果Java代码实现,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-02-02
  • Java Map集合详解与演示

    Java Map集合详解与演示

    Map用于保存具有映射关系的数据,Map集合里保存着两组值,一组用于保存Map的ley,另一组保存着Map的value,可以理解为Map中的元素是两个对象,一个对象作为键,一个对象作为值。键不可以重复,但是值可以重复
    2021-11-11
  • 从内存模型中了解Java final的全部细节

    从内存模型中了解Java final的全部细节

    关于final关键字,它也是我们一个经常用的关键字,可以修饰在类上、或者修饰在变量、方法上,以此看来定义它的一些不可变性!像我们经常使用的String类中,它便是final来修饰的类,并且它的字符数组也是被final所修饰的。但是一些final的一些细节你真的了解过吗
    2022-03-03
  • hadoop 切片机制分析与应用

    hadoop 切片机制分析与应用

    切片这个词对于做过python开发的同学一定不陌生,但是与hadoop中的切片有所区别,hadoop中的切片是为了优化hadoop的job在处理过程中MapTask阶段的性能达到最优而言
    2022-02-02
  • SpringBoot在线代码修改器的问题及解决方法

    SpringBoot在线代码修改器的问题及解决方法

    这篇文章主要介绍了SpringBoot在线代码修改器的问题及解决方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-06-06
  • Kotlin 基础教程之泛型

    Kotlin 基础教程之泛型

    这篇文章主要介绍了Kotlin 基础教程之泛型的相关资料,需要的朋友可以参考下
    2017-06-06
  • java中JsonObject与JsonArray转换方法实例

    java中JsonObject与JsonArray转换方法实例

    在项目日常开发中常常会遇到JSONArray和JSONObject的转换,很多公司刚入职的小萌新会卡在这里,下面这篇文章主要给大家介绍了关于java中JsonObject与JsonArray转换方法的相关资料,需要的朋友可以参考下
    2023-04-04
  • SpringBoot整合SpringDataRedis的示例代码

    SpringBoot整合SpringDataRedis的示例代码

    这篇文章主要介绍了SpringBoot整合SpringDataRedis的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-05-05
  • MyBatis ResultMap 的基本用法示例详解

    MyBatis ResultMap 的基本用法示例详解

    在MyBatis 中,resultMap 用于定义数据库查询结果到Java对象属性的映射关系,本文给大家介绍MyBatis ResultMap的基本用法,感兴趣的朋友一起看看吧
    2025-06-06
  • 详解SpringBoot中使用JPA作为数据持久化框架

    详解SpringBoot中使用JPA作为数据持久化框架

    这篇文章主要介绍了SpringBoot中使用JPA作为数据持久化框架的相关知识,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-03-03

最新评论