❝
在并发编程的世界里,我们经常会听说“锁”,这些锁的作用就像是我们家门前的门锁一样,保证同一时间只有一个人(或线程)能进入。在 Java 的多线程环境中,有两种“特别”的锁,它们分别是公平锁和非公平锁。本文将以通俗的语言和实用的示例代码对这两种锁进行讲解。
公平锁:排队买票的礼仪小士兵
想象一下你正在排队买电影票,突然有人插队,这让等了很久的你非常不公平。在 Java 的世界中,公平锁就是维护这种等待队列的一种锁。如果一个锁是公平的,那么它会按照线程请求的顺序来分配锁,保证“先来后到”。
公平锁的特征:
-
严格按请求顺序分配资源。 -
保证了等待时间最长的线程最先获得锁。 -
防止了线程饥饿(长时间得不到执行)。
使用公平锁的时候:
-
当你在乎每个线程都要公平获得资源时。 -
想要避免线程因为等待时间过长而处于饥饿状态。
非公平锁:无序世界中的闪电侠
再想象另外一种场景,你去超市买东西,虽然也是排队结账,但有时能够找到一个短队伍或者空闲的收银台,从而快速结账离开。在 Java 中,非公平锁就像是让线程抢占这个空闲收银台的机会。
非公平锁的特征:
-
可能不按请求顺序分配资源。 -
性能较好,因为不需要维护一个等待队列。 -
但有可能导致一些线程等待时间过长,即出现线程饥饿。
使用非公平锁的时候:
-
当你更在乎整体性能,而不太在乎部分线程等待时间。 -
当系统中有大量短小任务需要迅速处理时。
代码示例:
下面是一个使用ReentrantLock
的例子,实现了一个简单的服务,这里分别展示了如何实现公平锁和非公平锁。
公平锁的实现:
import java.util.concurrent.locks.ReentrantLock;
import org.springframework.stereotype.Service;
@Service
public class FairLockService {
private final ReentrantLock fairLock = new ReentrantLock(true); // 注意这里传入true表示公平锁
public void performService() {
fairLock.lock(); // 获取锁
try {
// 执行任务...
System.out.println("服务正在由线程处理:" + Thread.currentThread().getName());
} finally {
fairLock.unlock(); // 一定要在finally中释放锁,确保锁一定会被释放
}
}
}
非公平锁的实现:
import java.util.concurrent.locks.ReentrantLock;
import org.springframework.stereotype.Service;
@Service
public class UnfairLockService {
private final ReentrantLock unfairLock = new ReentrantLock(); // 默认是非公平锁
public void performService() {
unfairLock.lock(); // 获取锁
try {
// 执行任务...
System.out.println("服务正在由线程处理:" + Thread.currentThread().getName());
} finally {
unfairLock.unlock(); // 一定要在finally中释放锁
}
}
}
结语
公平锁和非公平锁就像世界上不同的管理资源的方法。对于 Java 程序来说,我们需要根据自己的需要选择使用哪一种。就像现实生活中,有时我们需要排队等待,而有时我们又希望快速完成自己的任务。
个人观点,仅供参考。
原文始发于微信公众号(源梦倩影):公平锁与非公平锁【通俗易懂篇】
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/291726.html