网站首页 > 文章精选 正文
lock 和 synchronized 都是 Java 中用来解决线程同步的机制,但是它们有以下几个区别:
1. lock 是一个接口,synchronized 是 Java 中的关键字,synchronized 内置于 Java 中,lock 需要额外导入
java.util.concurrent.locks 包。
2. lock 需要手动获取和释放锁,这可能导致死锁。而synchronized 会自动释放锁,所以不会出现忘记释放锁导致的死锁问题。
例如以下代码:
使用 lock:
Lock lock = new ReentrantLock();
lock.lock();
// do something...
// 如果这里忘记调用 lock.unlock() 释放锁,就会发生死锁
使用 synchronized :
public void method() {
synchronized (this) {
// do something...
}
}
// 这里不需要手动释放锁,JVM 会自动释放
3. lock 可以尝试获取锁,而synchronized 是直接获取锁,或者等待获取锁。例如:
if (lock.tryLock()) {
try {
// manipulate shared resource
} finally {
lock.unlock();
}
} else {
// do something else
}
4. lock可以具有不同的锁定机制(公平锁/非公平锁),而synchronized只有非公平锁。
5. synchronized 可重入锁,同一个线程外层函数获得锁之后,内层递归函数仍然能获取该锁。ReentrantLock 也可以是可重入的,需要在创建时设置参数为 true。
6. synchronized 无法知道目前有没有锁定和正在等待锁定的线程,ReentrantLock可以获取这些信息。
例如:
- lock.isLocked() - 检查锁是否被任何线程占用
- lock.getHoldCount() - 获取锁被当前线程重入的次数
- lock.hasQueuedThreads() - 检查是否有线程正在等待获得锁
- lock.getQueueLength() - 获取正在等待获得锁的线程数量示例代码:
Lock lock = new ReentrantLock();
// 检查锁是否被占用
boolean locked = lock.isLocked();
// 获取重入次数
int holdCount = lock.getHoldCount();
// 检查是否有等待线程
boolean hasQueued = lock.hasQueuedThreads();
// 获取等待线程数量
int queueLength = lock.getQueueLength();
而使用 synchronized 的话,没有这些方法可以获取锁的相关信息:
public synchronized void method() {
// 没有对应的方法获取锁信息
}
所以,总体来说,lock比synchronized更灵活方便,但是synchronized由JVM内置实现,对线程安全提供了更强的保证,且避免了手动获取和释放锁可能导致的问题,所以也更简便。选择哪个还是需要根据具体情况而定。
猜你喜欢
- 2025-07-10 Java基础——Java多线程(Lock接口详解)
- 2025-07-10 高并发环境下诡异的加锁问题(你加的锁未必安全)
- 2025-07-10 你真正了解synchronized关键字吗?
- 2025-07-10 Java 并发之 ReentrantReadWriteLock 深入分析
- 2025-07-10 ReentrantLock源码解析:ReentrantLock 的实现原理与 AQS 机制
- 2025-07-10 「Java多线程」内置锁(Synchronized)的前世今生
- 2025-07-10 面试:如何保证接口的幂等性?常见的实现方案有哪些?
- 2025-07-10 聊聊并发编程: Lock(并发编程的三大特性)
- 2025-07-10 Java并发之旅:Lock, Condition & ReadWriteLock 的魔法
- 2025-07-10 对volatile,synchronized,AQS的加锁解锁原理的一些理解
- 最近发表
-
- Vue3+Django4全新技术实战全栈项目|高清完结
- 工厂模式+策略模式消除 if else 实战
- 每天一个 Python 库:httpx异步请求,让接口测试飞起来
- 如何高效实现API接口的自动化测试?
- 前端工程化:从“手忙脚乱”到“从容协作”的进化记
- 使用C#创建服务端Web API(c#开发web服务器)
- SpringBoot之旅第四篇-web开发(springboot做web项目)
- 一文读懂SpringMVC(一文读懂新型政策性金融工具)
- Rust Web编程:第十二章 在 Rocket 中重新创建我们的应用程序
- Apache Druid 数据摄取——本地数据和kafka流式数据 一篇文章看懂
- 标签列表
-
- newcoder (56)
- 字符串的长度是指 (45)
- drawcontours()参数说明 (60)
- unsignedshortint (59)
- postman并发请求 (47)
- python列表删除 (50)
- 左程云什么水平 (56)
- 计算机网络的拓扑结构是指() (45)
- 编程题 (64)
- postgresql默认端口 (66)
- 数据库的概念模型独立于 (48)
- 产生系统死锁的原因可能是由于 (51)
- 数据库中只存放视图的 (62)
- 在vi中退出不保存的命令是 (53)
- 哪个命令可以将普通用户转换成超级用户 (49)
- noscript标签的作用 (48)
- 联合利华网申 (49)
- swagger和postman (46)
- 结构化程序设计主要强调 (53)
- 172.1 (57)
- apipostwebsocket (47)
- 唯品会后台 (61)
- 简历助手 (56)
- offshow (61)
- mysql数据库面试题 (57)