独家报道 lock.lock() 写在 try 外面?

开发 前端
ReentrantLock 是 JDK 提供的可重入的锁。提供对 共享资源的独占访问,一次只能有一个线程可以获取该锁。

[[378852]]

本文转载自微信公众号「源码兴趣圈」,作者龙台。转载本文请联系源码兴趣圈公众号。

前言

面试官:小伙子,JUC 并发包下的可重入锁 ReentrantLock 在代码里实际使用过么

混子:用过,ReentrantLock 是 JDK 提供的可重入的锁。提供对 共享资源的独占访问,一次只能有一个线程可以获取该锁

面试官:你觉得,ReentrantLock#lock 方法写到 try 语句外面还是里面

混子:我......

面试官:我们不合适,你走吧

先给出结论,lock.lock() 最规范的写法是写到 try 语句的外面

lock.lock()

Oracle 文档中在介绍锁的使用时有一段代码,我们以 ReentrantLock 举例,代码如下所示:

  1. ReentrantLock lock = new ReentrantLock(); 
  2. lock.lock(); 
  3. try { 
  4.     // access the resource protected by this lock 
  5. } finally { 
  6.     lock.unlock(); 

Q:为什么要把 lock.unlock() 放到 finally 语句块?

A:为了保证当前线程执行过程中出现异常时,锁依然能被释放掉,避免死锁的产生

我们来改动一下上面的代码,看看会产生什么样的影响

  1. ReentrantLock lock = new ReentrantLock(); 
  2. try { 
  3.     lock.lock(); 
  4.     // access the resource protected by this lock 
  5. } finally { 
  6.     lock.unlock(); 

看着没问题呀,为啥文章开始不建议这么用?先说下可能会存在的问题

异常堆栈丢失

假设在 lock.lock 方法中加锁异常(千万不要杠),那么会进入 finally 语句块中进行解锁

继续跟进,看一下 lock.unlock() 源码中是如何处理的

lock.lock() 抛出异常有可能还没获取到锁,那么 解锁源码中将当前线程比较拥有锁线程肯定是不相等的,所以会抛出 IMSE (IllegalMonitorStateException)异常

我重写了 ReentrantLock 加锁代码的逻辑,在里面抛出了异常,一起看下会出现什么情况

  1. final void lock() { 
  2.     // 模拟加锁未成功就抛出异常 
  3.     if (true) { 
  4.         throw new RuntimeException("报错啦!!!"); 
  5.     } 
  6.     if (compareAndSetState(0, 1)) 
  7.         setExclusiveOwnerThread(Thread.currentThread()); 
  8.     else 
  9.         acquire(1); 

根据下图可以看出 加锁时异常堆栈被 "吞掉了",悄无声息的就没了。当然这只是举例,但是谁能保证加锁未成功时不会抛出异常呢

真实存在的 BUG

上面代码示例中都是在 try 的第一行写 lock,出现问题的可能性极低。这里给大家提供一个反面教材,千万千万不要有这种类似行为

示例代码中把 lock 放到了 try 语句块里,然后 lock 加锁前面还有可能会产生异常的代码,这种就凉了,谁用谁凉的那种

结尾

所以关于要不要把 lock.lock() 写到 try 语句块里,文章的结论是:

最好是把 lock.lock() 加锁方法写到 try 外面,这是一种规范,而不是强制

如果你非要写到 try 里面,那么 请写到 try 语句块的第一行,或者 lock 加锁方法前面不会存在可能出现异常的代码

最后,如果你代码中加锁放到了 try 语句里,麻烦参考第 1 点

 

责任编辑:武晓燕 来源: 源码兴趣圈
相关推荐

2013-04-19 10:43:47

ChinaHadoop

2011-11-01 12:48:51

2011-10-27 15:28:09

Citrix思杰Synergy

2009-10-16 14:27:16

甲骨文虚拟化linux

2024-06-12 14:03:31

MySQLInnoDB

2011-08-24 16:41:38

LOCK中文man

2009-08-06 10:35:27

C# lock thi

2023-07-06 08:06:47

LockCondition公平锁

2010-10-14 17:17:07

软考2010年下半年软考试题

2011-12-12 19:45:19

IBM

2024-01-02 14:17:31

MySQLMDL LOCK语句

2017-02-14 10:00:19

Java开发Lock

2021-01-28 23:26:55

MySQL

2015-09-06 08:59:52

Java延时实例

2009-08-13 13:04:29

C# lock关键字

2009-08-26 15:16:29

C# lock关键字

2010-05-24 10:45:52

子命令Svn lock

2011-11-28 12:55:37

JavaJVM

2024-11-14 11:00:05

GolangGo结构体

2009-11-16 19:11:29

HPCxeon高性能计算
点赞
收藏

51CTO技术栈公众号