独家报道 lock.lock() 写在 try 外面?
本文转载自微信公众号「源码兴趣圈」,作者龙台。转载本文请联系源码兴趣圈公众号。
前言
面试官:小伙子,JUC 并发包下的可重入锁 ReentrantLock 在代码里实际使用过么
混子:用过,ReentrantLock 是 JDK 提供的可重入的锁。提供对 共享资源的独占访问,一次只能有一个线程可以获取该锁
面试官:你觉得,ReentrantLock#lock 方法写到 try 语句外面还是里面
混子:我......
面试官:我们不合适,你走吧
先给出结论,lock.lock() 最规范的写法是写到 try 语句的外面
lock.lock()
Oracle 文档中在介绍锁的使用时有一段代码,我们以 ReentrantLock 举例,代码如下所示:
- ReentrantLock lock = new ReentrantLock();
- lock.lock();
- try {
- // access the resource protected by this lock
- } finally {
- lock.unlock();
- }
Q:为什么要把 lock.unlock() 放到 finally 语句块?
A:为了保证当前线程执行过程中出现异常时,锁依然能被释放掉,避免死锁的产生
我们来改动一下上面的代码,看看会产生什么样的影响
- ReentrantLock lock = new ReentrantLock();
- try {
- lock.lock();
- // access the resource protected by this lock
- } finally {
- lock.unlock();
- }
看着没问题呀,为啥文章开始不建议这么用?先说下可能会存在的问题
异常堆栈丢失
假设在 lock.lock 方法中加锁异常(千万不要杠),那么会进入 finally 语句块中进行解锁
继续跟进,看一下 lock.unlock() 源码中是如何处理的
lock.lock() 抛出异常有可能还没获取到锁,那么 解锁源码中将当前线程比较拥有锁线程肯定是不相等的,所以会抛出 IMSE (IllegalMonitorStateException)异常
我重写了 ReentrantLock 加锁代码的逻辑,在里面抛出了异常,一起看下会出现什么情况
- final void lock() {
- // 模拟加锁未成功就抛出异常
- if (true) {
- throw new RuntimeException("报错啦!!!");
- }
- if (compareAndSetState(0, 1))
- setExclusiveOwnerThread(Thread.currentThread());
- else
- acquire(1);
- }
根据下图可以看出 加锁时异常堆栈被 "吞掉了",悄无声息的就没了。当然这只是举例,但是谁能保证加锁未成功时不会抛出异常呢
真实存在的 BUG
上面代码示例中都是在 try 的第一行写 lock,出现问题的可能性极低。这里给大家提供一个反面教材,千万千万不要有这种类似行为
示例代码中把 lock 放到了 try 语句块里,然后 lock 加锁前面还有可能会产生异常的代码,这种就凉了,谁用谁凉的那种
结尾
所以关于要不要把 lock.lock() 写到 try 语句块里,文章的结论是:
最好是把 lock.lock() 加锁方法写到 try 外面,这是一种规范,而不是强制
如果你非要写到 try 里面,那么 请写到 try 语句块的第一行,或者 lock 加锁方法前面不会存在可能出现异常的代码
最后,如果你代码中加锁放到了 try 语句里,麻烦参考第 1 点
相关文章
- 在 Go 里用 CGO?这 7 个问题你要关注!
- 9款优秀的去中心化通讯软件 Matrix 的客户端
- 求职数据分析,项目经验该怎么写
- 在OKR中,我看到了数据驱动业务的未来
- 火山引擎云原生大数据在金融行业的实践
- OpenHarmony富设备移植指南(二)—从postmarketOS获取移植资源
- 《数据成熟度指数》报告:64%的企业领袖认为大多数员工“不懂数据”
- OpenHarmony 小型系统兼容性测试指南
- 肯睿中国(Cloudera):2023年企业数字战略三大趋势预测
- 适用于 Linux 的十大命令行游戏
- GNOME 截图工具的新旧截图方式
- System76 即将推出的 COSMIC 桌面正在酝酿大变化
- 2GB 内存 8GB 存储即可流畅运行,Windows 11 极致精简版系统 Tiny11 发布
- 迎接 ecode:一个即将推出的具有全新图形用户界面框架的现代、轻量级代码编辑器
- loongarch架构介绍(三)—地址翻译
- Go 语言怎么解决编译器错误“err is shadowed during return”?
- 敏捷:可能被开发人员遗忘的部分
- Denodo预测2023年数据管理和分析的未来
- 利用数据推动可持续发展
- 在 Vue3 中实现 React 原生 Hooks(useState、useEffect),深入理解 React Hooks 的