IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Java 关于 Lock#lock() 的加锁位置 -> 正文阅读

[Java知识库]Java 关于 Lock#lock() 的加锁位置

1. Lock#lock() 的加锁位置问题

最近在做项目的性能优化,需要将原本单线程跑的程序改造成多线程并行以提高性能。然而业务资源池子是定量的,多线程并行势必涉及到共享资源抢占的问题,需要实现线程间的互斥等待。这种需求采用同步锁是毋庸置疑的,但是在加锁的位置上却有一些细节,例如加锁操作是否可以放在 try 代码块里面呢?

2. Lock#lock() 加锁位置分析

先给出结论,加锁操作推荐放在 try 代码块外部第一行,以下是 JDK 文档 给出的 Lock 使用示例,可以注意到两点:

  1. 加锁操作 Lock#lock() 方法调用放在 try 代码块外部第一行
  2. 解锁操作 Lock#unlock() 方法放在 finally 代码块内

第 2 点没什么好说的,解锁操作在 finally 里保证业务代码出现异常时锁依然能被释放,可以避免死锁产生。关于第 1 点则需要一些澄清,下文将解释以下两种情况可能产生的问题:

  1. 在 try 内部加锁
  2. 在 try 外部非第一行加锁
 Lock l = ...;
 l.lock();
 try {
   // access the resource protected by this lock
 } finally {
   l.unlock();
 }

2.1 加锁在 try 内部可能的问题

以下是将加锁操作放在 try 内部的代码示例,仔细考虑就能发现问题点:

  1. 假设 Lock#lock() 方法抛出运行时异常,如果加锁放在 try 代码块内部,则必然触发 finally 中的解锁方法 Lock#lunock() 的执行
  2. Lock#lunock() 方法会调用 AQS 的 tryRelease() 方法,如果当前 Lock 的实现类为独占锁(例如 ReentrantLock),那么其内部实现会检查当前解锁线程是否是持有锁的线程,如果不是则抛出 IllegalMonitorStateException 异常
  3. 当前线程在加锁方法 Lock#lock() 中抛出异常,显然当前线程未持有锁,然而它却在 finally 中进行解锁操作,通常情况下这里都会抛出一个解锁失败的 IllegalMonitorStateException 异常
  4. 以上过程中总共产生了两个异常,一个是在加锁时,另一个是在解锁时。然而在控制台查看异常堆栈,会发现加锁时异常堆栈丢失了,只剩下解锁时的异常堆栈,不利于问题排查
    Exception in thread "main" java.lang.IllegalMonitorStateException
        at java.util.concurrent.locks.ReentrantLock$Sync.tryRelease(ReentrantLock.java:151)
        at java.util.concurrent.locks.AbstractQueuedSynchronizer.release(AbstractQueuedSynchronizer.java:1261)
        at java.util.concurrent.locks.ReentrantLock.unlock(ReentrantLock.java:457)
    

以上问题的核心点在于,JDK 的独占锁 Lock 实现在解锁时会对操作线程进行校验,未持有锁的线程进行解锁操作将导致异常

    public static void main(String[] args) {
        Lock lock = new ReentrantLock();
        try {
            // 可在 Lock#lock() 方法执行前抛一个运行时异常,将抛异常和 Lock#lock() 方法执行
            // 整体当成一个拿锁的原子操作,模拟线程拿锁过程发生异常的情况
            // if (args.length == 0) throw new RuntimeException();
            lock.lock();
        } finally {
            lock.unlock();
        }
    }

2.2 加锁在 try 外部非第一行可能的问题

以下是加锁操作在 try 代码块外部非第一行的示例,简单解释下代码执行过程中可能产生的问题:

  1. 当前线程执行 Lock#lock() 方法正常拿到锁,继续往下执行
  2. 此处抛一个运行时异常模拟代码执行异常的情况,异常被抛出后当前线程执行中断,而此时线程还没有进入 try 代码块,finally 代码块必然不会执行,解锁操作自然更无从谈起
  3. 当前线程拿到锁后发生异常,终止执行却没有解锁,毫无疑问死锁就这样产生了
    public static void main(String[] args) {
        Lock lock = new ReentrantLock();
        lock.lock();
        if (args.length == 0) throw new RuntimeException();
        try {
            // TODO 
        } finally {
            lock.unlock();
        }
    }
  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-06-16 21:31:54  更:2022-06-16 21:33:26 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 19:21:47-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码