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实现原理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot反射的基本应用全解析

    SpringBoot反射的基本应用全解析

    反射是Java的核心特性之一,允许在运行时检查或修改类、方法、字段的行为,SpringBoot作为基于Spring框架的快速开发工具,广泛利用反射实现依赖注入、动态代理等功能,本文给大家介绍SpringBoot反射应用全解析,感兴趣的朋友跟随小编一起看看吧
    2025-09-09
  • Java并发编程this逃逸问题总结

    Java并发编程this逃逸问题总结

    本篇文章给大家详细分析了Java并发编程this逃逸的问题分享,对此有需要的朋友参考下。
    2018-02-02
  • SpringMVC @RequestBody的使用解析

    SpringMVC @RequestBody的使用解析

    这篇文章主要介绍了SpringMVC @RequestBody的使用解析,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • Java实现一个小说采集程序的简单实例

    Java实现一个小说采集程序的简单实例

    下面小编就为大家带来一篇Java实现一个小说采集程序的简单实例。小编觉得挺不错的, 现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-06-06
  • Java中十进制和十六进制的相互转换方法

    Java中十进制和十六进制的相互转换方法

    下面小编就为大家带来一篇Java中十进制和十六进制的相互转换方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-08-08
  • Java实现XML与JSON秒级转换示例详解

    Java实现XML与JSON秒级转换示例详解

    这篇文章主要为大家介绍了Java实现XML与JSON秒级转换示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-09-09
  • Java多线程开发之并发安全详解

    Java多线程开发之并发安全详解

    这篇文章主要介绍了Java多线程开发之并发安全,并发安全是指在多个线程同时访问共享资源时,通过锁、volatile、原子类或线程安全集合等机制,防止数据竞争和不一致,确保程序执行的正确性和稳定性,需要的朋友可以参考下
    2026-01-01
  • java中URLEncoder.encode与URLDecoder.decode处理url特殊参数的方法

    java中URLEncoder.encode与URLDecoder.decode处理url特殊参数的方法

    这篇文章主要给大家介绍了关于java中URLEncoder.encode与URLDecoder.decode处理url特殊参数的方法,文中介绍的非常详细,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-03-03
  • SpringBoot处理请求参数中包含特殊符号

    SpringBoot处理请求参数中包含特殊符号

    今天写代码遇到了一个问题,请求参数是个路径“D:/ExcelFile”,本文就详细的介绍一下该错误的解决方法,感兴趣的可以了解一下
    2021-06-06
  • Java实现对两个List快速去重并排序操作示例

    Java实现对两个List快速去重并排序操作示例

    这篇文章主要介绍了Java实现对两个List快速去重并排序操作,结合实例形式较为详细的分析了Java针对list的遍历、去重、排序相关操作技巧与注意事项,需要的朋友可以参考下
    2018-07-07

最新评论