interface Lock {
void lock();
void unlock() throws Exception;
}
class CLHLock implements Lock {
/**
* tailNode 尾节点原子操作保证线程安全
*/
private final AtomicReference<Node> tailNode = new AtomicReference<>();
private final ThreadLocal<Node> currentNodeLocal = new ThreadLocal<>();
private static class Node {
/**
* 前驱节点
*/
private Node preNode;
/**
* 当前节点状态
* volatile 保证对后置线程的可见性
*/
private volatile Boolean lockState;
public Node(Boolean lockState) {
this.lockState = lockState;
}
}
@Override
public void lock() {
Node currentNode = currentNodeLocal.get();
if (currentNode == null) {
currentNodeLocal.set(new Node(true));
currentNode = currentNodeLocal.get();
}
// 拿到当前节点的前置节点 形成逻辑连接 无实际连接
Node preNode = tailNode.getAndSet(currentNode);
// 检查前置节点 lock state
while (currentNode.preNode != null && preNode.lockState) {
System.out.println(Thread.currentThread().getName() + " 自旋等待获取锁");
}
System.out.println(Thread.currentThread().getName() + " 获取锁成功");
}
@Override
public void unlock() throws Exception {
Node currentNode = currentNodeLocal.get();
if (!currentNode.lockState || (currentNode.preNode != null && currentNode.preNode.lockState)) {
throw new Exception("current thread is not locked");
}
currentNode.lockState = false;
// 清除线程 ThreadLocal 本次锁信息 避免拿到已经释放的锁信息
currentNodeLocal.remove();
System.out.println(Thread.currentThread().getName() + " 释放锁");
}
}
interface Lock {
void lock();
void unlock() throws Exception;
}
class MSCLock implements Lock {
/**
* tailNode 尾节点原子操作保证线程安全
*/
final AtomicReference<Node> tailNode = new AtomicReference<>();
private final ThreadLocal<Node> currentNodeLocal = new ThreadLocal<>();
private static class Node {
/**
* 后驱节点
* volatile 保证 nextNode 引用的可见性
*/
private volatile Node nextNode;
/**
* 当前节点状态
* volatile 保证对后置线程的可见性
*/
private volatile Boolean lockState;
public Node(Boolean lockState) {
this.lockState = lockState;
}
}
@Override
public void lock() {
Node currentNode = new Node(true);
currentNodeLocal.set(currentNode);
Node preNode = tailNode.getAndSet(currentNode);
// 首节点直接获取锁
if (preNode == null) {
currentNode.lockState = true;
} else {
preNode.nextNode = currentNode;
// 自旋检测当前节点状态
while (!currentNode.lockState) {
System.out.println(Thread.currentThread().getName() + " 自旋等待获取锁");
}
}
System.out.println(Thread.currentThread().getName() + " 获取锁成功");
}
@Override
public void unlock() {
Node currentNode = currentNodeLocal.get();
Node nextNode = currentNode.nextNode;
// 若无等待线程 尝试将tailNode置为 null
if (nextNode == null) {
if (tailNode.compareAndSet(currentNode, null)) {
System.out.println(Thread.currentThread().getName() + " 锁释放成功");
return;
} else {
nextNode = currentNode.nextNode;
}
}
// 清除线程 ThreadLocal 本次锁信息 避免拿到已经释放的锁信息
currentNodeLocal.remove();
// 唤醒下一个等待线程
nextNode.lockState = true;
}
}
👋 你好,我是 Lorin 洛林,一位 Java 后端技术开发者!座右铭:Technology has the power to make the world a better place.
🚀 我对技术的热情是我不断学习和分享的动力。我的博客是一个关于Java生态系统、后端开发和最新技术趋势的地方。
🧠 作为一个 Java 后端技术爱好者,我不仅热衷于探索语言的新特性和技术的深度,还热衷于分享我的见解和最佳实践。我相信知识的分享和社区合作可以帮助我们共同成长。
💡 在我的博客上,你将找到关于Java核心概念、JVM 底层技术、常用框架如Spring和Mybatis 、MySQL等数据库管理、RabbitMQ、Rocketmq等消息中间件、性能优化等内容的深入文章。我也将分享一些编程技巧和解决问题的方法,以帮助你更好地掌握Java编程。
🌐 我鼓励互动和建立社区,因此请留下你的问题、建议或主题请求,让我知道你感兴趣的内容。此外,我将分享最新的互联网和技术资讯,以确保你与技术世界的最新发展保持联系。我期待与你一起在技术之路上前进,一起探讨技术世界的无限可能性。
📖 保持关注我的博客,让我们共同追求技术卓越。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。