Java并发编程-重入锁

简介: 章节目录什么是重入锁底层实现-如何实现重入公平与非公平获取锁的区别与底层实现1.什么是重入锁1.1 重入锁的定义重入锁ReentrantLock,支持重入的锁,表示一个线程对资源的重复加锁。

章节目录

  • 什么是重入锁
  • 底层实现-如何实现重入
  • 公平与非公平获取锁的区别与底层实现

1.什么是重入锁

1.1 重入锁的定义

重入锁ReentrantLock,支持重入的锁,表示一个线程对资源的重复加锁。

1.2 重入锁的特性

1.重进入
2.非/公平性获取锁

1.3 自定义同步器Mutex 的缺陷

当线程调用Mutex的lock()方法获取锁之后,再次调用lock()方法,该线程将会被
自己阻塞,原因是Mutex在实现tryAcquire(int acquires)方法时没有考虑占有锁
的线程再次获取锁的场景。

1.4 ReentrantLock & synchronized 关键字

1.synchronized 关键字支持隐式的重进入
2.ReentrantLock 在调用lock() 方法时,已经获取到锁的线程,能够再次调用
lock()方法获取到锁而不被阻塞,即可支持重入

1.4 公平性获取锁

公平性 含义
公平性获取锁 在绝对时间上,先对锁进行获取请求的请求一定先被满足,那么这个锁就是公平的
非公平性获取锁 无上述限制

事实上 公平锁机制往往没有非公平性机制获取锁的效率高,因为会牵扯到频繁的上下文切换,但公平锁可以减少饥饿发生的概率,等待越久的请求越能得到优先满足。

2. 底层实现-如何实现重入

重进入是指任意线程在获取到锁之后能够再次获取该锁,而不被阻塞,改特性实现需要解决以下两个问题:

  • 线程再次获取锁
    线程再次获取锁。锁需要去识别获取锁的线程是否为当前占据锁的线程,如果是,则再次成功获取。
    
  • 锁的最终释放
    线程重复n次获取了锁,随后在第n次释放锁,锁的释放要求锁对于被获取递
    增的次数进行递减操作,当计数==0时表示锁已经成功释放。
    

2.1 可重入锁的源码
非公平性获取同步状态(锁)的 nonfairTryAcquire() 方法

    final boolean nonfairTryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            int c = getState();
            if (c == 0) {
                if (compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0) // overflow
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }

该方法增加了再次获取同步状态的处理逻辑:通过判断当前线程是否为获取锁的线程来决定获取操作是否成功,如果是获取锁的线程的再次请求 则将同步状态值计数器进行递增并返回true,表示获取同步状态成功。

释放同步状态(锁)

protected final boolean tryRelease(int releases) {
            int c = getState() - releases;
            if (Thread.currentThread() != getExclusiveOwnerThread())
                throw new IllegalMonitorStateException();
            boolean free = false;
            if (c == 0) {
                free = true;
                setExclusiveOwnerThread(null);
            }
            setState(c);
            return free;
        }

如果该锁被获取了n次,那么前n-1此tryRelease(int release) 方法必须返回false,而只有同步状态完全释放了,才能返回true。

3.公平与非公平获取锁的区别与底层实现

3.1 公平性获取锁的底层实现
公平性获取锁即按照客观时间顺序,FIFO方式获取同步状态
具体源码如下所示

  protected final boolean tryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            int c = getState();
            if (c == 0) {
                if (!hasQueuedPredecessors() &&
                    compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0)
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }

公平性获取同步状态的与非公平性获取同步状态的区别在于hasQueuedPredecessors()方法的使用,即加入了当前节点是否有前驱节点的判断,如果该方法返回true,则表示有线程比当前线程更早的加入到同步队列(更早的请求获取锁),因此需要等待前驱线程获取并释放锁之后才能继续获取锁。

非公平性获取锁的实现

  • 公平性获取锁保证了锁的获取顺序按照FIFO原则,不会出现线程“饥饿”的现象,但代价是进行大量的线程切换。
  • 非公平性锁虽然可能造成线程饥饿,但是有极少的线程切换,保证了其更大的吞吐量。
目录
相关文章
|
3天前
|
安全 Java API
JAVA并发编程JUC包之CAS原理
在JDK 1.5之后,Java API引入了`java.util.concurrent`包(简称JUC包),提供了多种并发工具类,如原子类`AtomicXX`、线程池`Executors`、信号量`Semaphore`、阻塞队列等。这些工具类简化了并发编程的复杂度。原子类`Atomic`尤其重要,它提供了线程安全的变量更新方法,支持整型、长整型、布尔型、数组及对象属性的原子修改。结合`volatile`关键字,可以实现多线程环境下共享变量的安全修改。
|
2天前
|
存储 Java
JAVA并发编程AQS原理剖析
很多小朋友面试时候,面试官考察并发编程部分,都会被问:说一下AQS原理。面对并发编程基础和面试经验,专栏采用通俗简洁无废话无八股文方式,已陆续梳理分享了《一文看懂全部锁机制》、《JUC包之CAS原理》、《volatile核心原理》、《synchronized全能王的原理》,希望可以帮到大家巩固相关核心技术原理。今天我们聊聊AQS....
|
2天前
|
Java 程序员 数据库连接
Java编程中的异常处理:从基础到进阶
【9月更文挑战第18天】在Java的世界里,异常处理是每个程序员必须面对的挑战。本文将带你从异常的基本概念出发,通过实际的代码示例,深入探讨如何有效地管理和处理异常。我们将一起学习如何使用try-catch块来捕捉异常,理解finally块的重要性,以及如何自定义异常类来满足特定需求。无论你是初学者还是有经验的开发者,这篇文章都将为你提供新的见解和技巧,让你的Java代码更加健壮和可靠。
|
2天前
|
Java 数据库连接 UED
掌握Java编程中的异常处理
【9月更文挑战第18天】在Java的世界中,异常是那些不请自来的客人,它们可能在任何时候突然造访。本文将带你走进Java的异常处理机制,学习如何优雅地应对这些突如其来的“访客”。从基本的try-catch语句到更复杂的自定义异常,我们将一步步深入,确保你能够在面对异常时,不仅能够从容应对,还能从中学到宝贵的经验。让我们一起探索如何在Java代码中实现健壮的异常处理策略,保证程序的稳定运行。
|
3天前
|
Java 数据库
JAVA并发编程-一文看懂全部锁机制
曾几何时,面试官问:java都有哪些锁?小白,一脸无辜:用过的有synchronized,其他不清楚。面试官:回去等通知! 今天我们庖丁解牛说说,各种锁有什么区别、什么场景可以用,通俗直白的分析,让小白再也不怕面试官八股文拷打。
|
3天前
|
Java
深入理解Java中的多线程编程
本文将探讨Java多线程编程的核心概念和技术,包括线程的创建与管理、同步机制以及并发工具类的应用。我们将通过实例分析,帮助读者更好地理解和应用Java多线程编程,提高程序的性能和响应能力。
15 4
|
3天前
|
安全 Java 开发者
Java并发编程中的锁机制解析
本文深入探讨了Java中用于管理多线程同步的关键工具——锁机制。通过分析synchronized关键字和ReentrantLock类等核心概念,揭示了它们在构建线程安全应用中的重要性。同时,文章还讨论了锁机制的高级特性,如公平性、类锁和对象锁的区别,以及锁的优化技术如锁粗化和锁消除。此外,指出了在高并发环境下锁竞争可能导致的问题,并提出了减少锁持有时间和使用无锁编程等策略来优化性能的建议。最后,强调了理解和正确使用Java锁机制对于开发高效、可靠并发应用程序的重要性。
13 3
|
2天前
|
安全 Java 调度
Java 并发编程中的线程安全和性能优化
本文将深入探讨Java并发编程中的关键概念,包括线程安全、同步机制以及性能优化。我们将从基础入手,逐步解析高级技术,并通过实例展示如何在实际开发中应用这些知识。阅读完本文后,读者将对如何在多线程环境中编写高效且安全的Java代码有一个全面的了解。
|
2天前
|
Java
JAVA并发编程ReentrantLock核心原理剖析
本文介绍了Java并发编程中ReentrantLock的重要性和优势,详细解析了其原理及源码实现。ReentrantLock作为一种可重入锁,弥补了synchronized的不足,如支持公平锁与非公平锁、响应中断等。文章通过源码分析,展示了ReentrantLock如何基于AQS实现公平锁和非公平锁,并解释了两者的具体实现过程。
|
2天前
|
Kubernetes Cloud Native Java
探索未来编程新纪元:Quarkus带你秒建高性能Kubernetes原生Java应用,云原生时代的技术狂欢!
Quarkus 是专为 Kubernetes 设计的全栈云原生 Java 框架,凭借其轻量级、快速启动及高效执行特性,在 Java 社区脱颖而出。通过编译时优化与原生镜像支持,Quarkus 提升了应用性能,同时保持了 Java 的熟悉度与灵活性。本文将指导你从创建项目、编写 REST 控制器到构建与部署 Kubernetes 原生镜像的全过程,让你快速上手 Quarkus,体验高效开发与部署的乐趣。
9 0