JUC小册——公平锁和非公平锁

本文目录

公平锁和非公平锁是锁的两种不同实现策略,主要用于多线程环境下对共享资源的访问控制,下面从定义、区别、优缺点、使用场景等方面详细介绍:

一、定义

  • 公平锁 :是指多个线程按照申请锁的顺序来获取锁,就像排队一样,先到先得。当一个线程请求锁时,如果锁当前被其他线程持有,那么该线程会被放入等待队列中,等持有锁的线程释放锁后,等待队列中最前面的线程会获得锁。在 Java 中,ReentrantLock可以通过构造函数ReentrantLock(boolean fair)并传入true来创建公平锁。示例代码如下:
java 复制代码
import java.util.concurrent.locks.ReentrantLock;

public class FairLock {
    private static final ReentrantLock fairLock = new ReentrantLock(true);

    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                fairLock.lock();
                try {
                    System.out.println(Thread.currentThread().getName() + " 获得锁");
                } finally {
                    fairLock.unlock();
                }
            }).start();
        }
    }
}
  • 非公平锁 :多个线程获取锁的顺序并不按照申请锁的顺序,当锁被释放时,任何一个等待锁的线程都有机会获得锁,而不是按照等待顺序。在 Java 中,synchronized关键字和ReentrantLock默认都是非公平锁。例如:
java 复制代码
import java.util.concurrent.locks.ReentrantLock;

public class NonFairLock {
    private static final ReentrantLock nonFairLock = new ReentrantLock();

    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                nonFairLock.lock();
                try {
                    System.out.println(Thread.currentThread().getName() + " 获得锁");
                } finally {
                    nonFairLock.unlock();
                }
            }).start();
        }
    }
}

二、两者区别

  • 获取锁的顺序 :公平锁严格按照线程请求锁的顺序来分配锁;非公平锁不保证线程获取锁的顺序,新请求锁的线程可能会在等待队列中的线程之前获得锁。
  • 实现机制 :公平锁在每次获取锁时,会先检查等待队列中是否有其他线程在等待,如果有则将自己放入等待队列尾部;非公平锁在锁被释放时,新请求锁的线程会直接尝试获取锁,而不考虑等待队列中是否有其他线程。

三、优缺点

  • 公平锁
    • 优点 :公平锁保证了线程获取锁的公平性,每个线程都有机会按照顺序获取锁,避免了某些线程长时间等待锁的情况,适用于对公平性要求较高的场景。
    • 缺点:由于需要维护等待队列,公平锁的实现相对复杂,性能开销较大,在高并发场景下可能会导致吞吐量下降。
  • 非公平锁
    • 优点:非公平锁的实现简单,性能较高,因为它减少了线程上下文切换和等待队列的维护开销。在高并发场景下,非公平锁可以提高系统的吞吐量。
    • 缺点:可能会导致某些线程长时间得不到锁,出现"饥饿"现象。

四、适合场景

  • 公平锁:适用于对公平性要求较高的场景,例如任务调度系统,每个任务都希望按照提交的顺序依次执行;或者资源分配系统,需要保证每个线程都能公平地获取资源。
  • 非公平锁 :适用于对性能要求较高、对公平性要求较低的场景,例如大多数的并发编程场景,如缓存系统的并发访问、数据库连接池的并发操作等,这些场景更注重系统的整体性能和吞吐量。

|-----------------------------------------------------------------------------------------------------|--------------------|--------------------------------------------------------------------------------------------------|
| ← 上一篇 Java进阶------常用类及常用方法详解 | 记得点赞、关注、收藏哦! | 下一篇 Java进阶------数组超详细整理 → |

相关推荐
wei_shuo21 分钟前
飞算 JavaAI 开发助手:深度学习驱动下的 Java 全链路智能开发新范式
java·开发语言·飞算javaai
熊猫钓鱼>_>21 分钟前
用Python解锁图像处理之力:从基础到智能应用的深度探索
开发语言·图像处理·python
GO兔32 分钟前
开篇:GORM入门——Go语言的ORM王者
开发语言·后端·golang·go
欧阳秦穆43 分钟前
apoc-5.24.0-extended.jar 和 apoc-4.4.0.36-all.jar 啥区别
java·jar
好开心啊没烦恼1 小时前
Python 数据分析:numpy,抽提,整数数组索引与基本索引扩展(元组传参)。听故事学知识点怎么这么容易?
开发语言·人工智能·python·数据挖掘·数据分析·numpy·pandas
岁忧1 小时前
(LeetCode 面试经典 150 题 ) 58. 最后一个单词的长度 (字符串)
java·c++·算法·leetcode·面试·go
Java初学者小白1 小时前
秋招Day14 - Redis - 应用
java·数据库·redis·缓存
代码老y1 小时前
Spring Boot + 本地部署大模型实现:优化与性能提升
java·spring boot·后端
GodKeyNet1 小时前
设计模式-桥接模式
java·设计模式·桥接模式
future14122 小时前
C#学习日记
开发语言·学习·c#