聊聊ReentrantLock 中的四个坑!

开发 前端
本文介绍了 Java 中的显式锁 Lock 及其子类 ReentrantLock 的使用和注意事项,Lock 在 Java 中占据了锁的半壁江山,但在使用时却要注意四个问题。

作者 | 王磊

来源 | Java中文社群(ID:javacn666)

转载请联系授权(微信ID:GG_Stone)

JDK 1.5 之前 synchronized 的性能是比较低的,但在 JDK 1.5 中,官方推出一个重量级功能 Lock,一举改变了 Java 中锁的格局。JDK 1.5 之前当我们谈到锁时,只能使用内置锁 synchronized,但如今我们锁的实现又多了一种显式锁 Lock。

本文咱们重点来看 Lock。

Lock 简介

Lock 是一个顶级接口,它的所有方法如下图所示:

它的子类列表如下:

 

我们通常会使用 ReentrantLock 来定义其实例,它们之间的关联如下图所示:

“PS:Sync 是同步锁的意思,FairSync 是公平锁,NonfairSync 是非公平锁。

ReentrantLock 使用

学习任何一项技能都是先从使用开始的,所以我们也不例外,咱们先来看下 ReentrantLock 的基础使用:

  1. publicclass LockExample { 
  2.     // 创建锁对象 
  3.     privatefinal ReentrantLock lock = new ReentrantLock(); 
  4.     public void method() { 
  5.         // 加锁操作 
  6.         lock.lock(); 
  7.         try { 
  8.             // 业务代码...... 
  9.         } finally { 
  10.             // 释放锁 
  11.             lock.unlock(); 
  12.         } 
  13.     } 

ReentrantLock 在创建之后,有两个关键性的操作:

  • 加锁操作:lock()
  • 释放锁操作:unlock()

ReentrantLock 中的坑

1.ReentrantLock 默认为非公平锁

很多人会认为(尤其是新手朋友),ReentrantLock 默认的实现是公平锁,其实并非如此,ReentrantLock 默认情况下为非公平锁(这主要是出于性能方面的考虑),比如下面这段代码:

  1. import java.util.concurrent.locks.ReentrantLock; 
  2.  
  3. publicclass LockExample { 
  4.     // 创建锁对象 
  5.     privatestaticfinal ReentrantLock lock = new ReentrantLock(); 
  6.  
  7.     public static void main(String[] args) { 
  8.         // 定义线程任务 
  9.         Runnable runnable = new Runnable() { 
  10.             @Override 
  11.             public void run() { 
  12.                 // 加锁 
  13.                 lock.lock(); 
  14.                 try { 
  15.                     // 打印执行线程的名字 
  16.                     System.out.println("线程:" + Thread.currentThread().getName()); 
  17.                 } finally { 
  18.                     // 释放锁 
  19.                     lock.unlock(); 
  20.                 } 
  21.             } 
  22.         }; 
  23.         // 创建多个线程 
  24.         for (int i = 0; i < 10; i++) { 
  25.             new Thread(runnable).start(); 
  26.         } 
  27.     } 

以上程序的执行结果如下:

从上述执行的结果可以看出,ReentrantLock 默认情况下为非公平锁。因为线程的名称是根据创建的先后顺序递增的,所以如果是公平锁,那么线程的执行应该是有序递增的,但从上述的结果可以看出,线程的执行和打印是无序的,这说明 ReentrantLock 默认情况下为非公平锁。

想要将 ReentrantLock 设置为公平锁也很简单,只需要在创建 ReentrantLock 时,设置一个 true 的构造参数就可以了,如下代码所示:

  1. import java.util.concurrent.locks.ReentrantLock; 
  2.  
  3. publicclass LockExample { 
  4.     // 创建锁对象(公平锁) 
  5.     privatestaticfinal ReentrantLock lock = new ReentrantLock(true); 
  6.  
  7.     public static void main(String[] args) { 
  8.         // 定义线程任务 
  9.         Runnable runnable = new Runnable() { 
  10.             @Override 
  11.             public void run() { 
  12.                 // 加锁 
  13.                 lock.lock(); 
  14.                 try { 
  15.                     // 打印执行线程的名字 
  16.                     System.out.println("线程:" + Thread.currentThread().getName()); 
  17.                 } finally { 
  18.                     // 释放锁 
  19.                     lock.unlock(); 
  20.                 } 
  21.             } 
  22.         }; 
  23.         // 创建多个线程 
  24.         for (int i = 0; i < 10; i++) { 
  25.             new Thread(runnable).start(); 
  26.         } 
  27.     } 

以上程序的执行结果如下:

从上述结果可以看出,当我们显式的给 ReentrantLock 设置了 true 的构造参数之后,ReentrantLock 就变成了公平锁,线程获取锁的顺序也变成有序的了。

其实从 ReentrantLock 的源码我们也可以看出它究竟是公平锁还是非公平锁,ReentrantLock 部分源码实现如下:

  1. public ReentrantLock() { 
  2.      sync = new NonfairSync(); 
  3.  } 
  4. public ReentrantLock(boolean fair) { 
  5.     sync = fair ? new FairSync() : new NonfairSync(); 

从上述源码中可以看出,默认情况下 ReentrantLock 会创建一个非公平锁,如果在创建时显式的设置构造参数的值为 true 时,它就会创建一个公平锁。

2.在 finally 中释放锁

使用 ReentrantLock 时一定要记得释放锁,否则就会导致该锁一直被占用,其他使用该锁的线程则会永久的等待下去,所以我们在使用 ReentrantLock 时,一定要在 finally 中释放锁,这样就可以保证锁一定会被释放。

反例

  1. import java.util.concurrent.locks.ReentrantLock; 
  2.  
  3. publicclass LockExample { 
  4.     // 创建锁对象 
  5.     privatestaticfinal ReentrantLock lock = new ReentrantLock(); 
  6.     public static void main(String[] args) { 
  7.         // 加锁操作 
  8.         lock.lock(); 
  9.         System.out.println("Hello,ReentrantLock."); 
  10.         // 此处会报异常,导致锁不能正常释放 
  11.         int number = 1 / 0; 
  12.         // 释放锁 
  13.         lock.unlock(); 
  14.         System.out.println("锁释放成功!"); 
  15.     } 

以上程序的执行结果如下:

从上述结果可以看出,当出现异常时锁未被正常释放,这样就会导致其他使用该锁的线程永久的处于等待状态。

正例

  1. import java.util.concurrent.locks.ReentrantLock; 
  2.  
  3. publicclass LockExample { 
  4.     // 创建锁对象 
  5.     privatestaticfinal ReentrantLock lock = new ReentrantLock(); 
  6.     public static void main(String[] args) { 
  7.         // 加锁操作 
  8.         lock.lock(); 
  9.         try { 
  10.             System.out.println("Hello,ReentrantLock."); 
  11.             // 此处会报异常 
  12.             int number = 1 / 0; 
  13.         } finally { 
  14.             // 释放锁 
  15.             lock.unlock(); 
  16.             System.out.println("锁释放成功!"); 
  17.         } 
  18.     } 

以上程序的执行结果如下:

从上述结果可以看出,虽然方法中出现了异常情况,但并不影响 ReentrantLock 锁的释放操作,这样其他使用此锁的线程就可以正常获取并运行了。

3.锁不能被释放多次

lock 操作的次数和 unlock 操作的次数必须一一对应,且不能出现一个锁被释放多次的情况,因为这样就会导致程序报错。

反例

一次 lock 对应了两次 unlock 操作,导致程序报错并终止执行,示例代码如下:

  1. import java.util.concurrent.locks.ReentrantLock; 
  2.  
  3. publicclass LockExample { 
  4.     // 创建锁对象 
  5.     privatestaticfinal ReentrantLock lock = new ReentrantLock(); 
  6.  
  7.     public static void main(String[] args) { 
  8.         // 加锁操作 
  9.         lock.lock(); 
  10.          
  11.         // 第一次释放锁 
  12.         try { 
  13.             System.out.println("执行业务 1~"); 
  14.             // 业务代码 1...... 
  15.         } finally { 
  16.             // 释放锁 
  17.             lock.unlock(); 
  18.             System.out.println("锁释锁"); 
  19.         } 
  20.  
  21.         // 第二次释放锁 
  22.         try { 
  23.             System.out.println("执行业务 2~"); 
  24.             // 业务代码 2...... 
  25.         } finally { 
  26.             // 释放锁 
  27.             lock.unlock(); 
  28.             System.out.println("锁释锁"); 
  29.         } 
  30.         // 最后的打印操作 
  31.         System.out.println("程序执行完成."); 
  32.     } 

以上程序的执行结果如下:

从上述结果可以看出,执行第 2 个 unlock 时,程序报错并终止执行了,导致异常之后的代码都未正常执行。

4.lock 不要放在 try 代码内

在使用 ReentrantLock 时,需要注意不要将加锁操作放在 try 代码中,这样会导致未加锁成功就执行了释放锁的操作,从而导致程序执行异常。

反例

  1. import java.util.concurrent.locks.ReentrantLock; 
  2.  
  3. publicclass LockExample { 
  4.     // 创建锁对象 
  5.     privatestaticfinal ReentrantLock lock = new ReentrantLock(); 
  6.  
  7.     public static void main(String[] args) { 
  8.         try { 
  9.             // 此处异常 
  10.             int num = 1 / 0; 
  11.             // 加锁操作 
  12.             lock.lock(); 
  13.         } finally { 
  14.             // 释放锁 
  15.             lock.unlock(); 
  16.             System.out.println("锁释锁"); 
  17.         } 
  18.         System.out.println("程序执行完成."); 
  19.     } 

以上程序的执行结果如下:

从上述结果可以看出,如果将加锁操作放在 try 代码中,可能会导致两个问题:

  1. 未加锁成功就执行了释放锁的操作,从而导致了新的异常;
  2. 释放锁的异常会覆盖程序原有的异常,从而增加了排查问题的难度。

总结

本文介绍了 Java 中的显式锁 Lock 及其子类 ReentrantLock 的使用和注意事项,Lock 在 Java 中占据了锁的半壁江山,但在使用时却要注意 4 个问题:

  1. 默认情况下 ReentrantLock 为非公平锁而非公平锁;
  2. 加锁次数和释放锁次数一定要保持一致,否则会导致线程阻塞或程序异常;
  3. 加锁操作一定要放在 try 代码之前,这样可以避免未加锁成功又释放锁的异常;
  4. 释放锁一定要放在 finally 中,否则会导致线程阻塞。

 

责任编辑:姜华 来源: Java中文社群
相关推荐

2021-11-05 07:59:25

HashMapJava知识总结

2022-01-12 15:50:24

JavaScript开发循环

2022-12-19 16:07:22

数据治理IT

2022-12-12 08:47:06

2020-08-13 10:29:55

项目管理项目经理CIO

2022-05-04 12:44:57

Python编程语言

2024-04-11 09:38:15

2022-03-02 08:20:54

并发编程java后端开发

2021-01-25 10:40:56

Python 开发编程语言

2011-07-14 15:23:34

java

2024-05-10 12:33:06

flask装饰器

2024-06-25 12:45:05

2013-03-18 13:31:28

2022-02-23 15:09:18

数字化转型国有企业数据

2010-03-30 11:00:46

Oracle 数据

2023-10-26 07:47:35

JavaScript代码变量

2023-08-21 13:39:57

开发桌面Ubuntu

2011-05-18 09:32:14

java

2024-11-14 09:00:00

Python编程元编程

2010-06-29 09:06:39

Java思想Java虚拟机
点赞
收藏

51CTO技术栈公众号