synchronized 和 Reentrantlock

多线程编程中,当代码需要同步时我们会用到锁。Java为我们提供了内置锁(synchronized)和显式锁(ReentrantLock)两种同步方式。显式锁是JDK1.5引入的,这两种锁有什么异同呢?是仅仅增加了一种选择还是另有其因?本文为您一探究竟。

// synchronized关键字用法示例

public synchronized void add(int t){// 同步方法

this.v += t;

}

public static synchronized void sub(int t){// 同步静态方法

value -= t;

}

public int decrementAndGet(){

synchronized(obj){// 同步代码块

return --v;

}

}

这就是内置锁的全部用法,你已经学会了。

内置锁使用起来非常方便,不需要显式的获取和释放,任何一个对象都能作为一把内置锁。使用内置锁能够解决大部分的同步场景。“任何一个对象都能作为一把内置锁”也意味着出现synchronized关键字的地方,都有一个对象与之关联,具体说来:

当synchronized作用于普通方法是,锁对象是this;

当synchronized作用于静态方法是,锁对象是当前类的Class对象;

当synchronized作用于代码块时,锁对象是synchronized(obj)中的这个obj。

显式锁

内置锁这么好用,为什么还需多出一个显式锁呢?因为有些事情内置锁是做不了的,比如:

我们想给锁加个等待时间超时时间,超时还未获得锁就放弃,不至于无限等下去;

我们想以可中断的方式获取锁,这样外部线程给我们发一个中断信号就能唤起等待锁的线程;

我们想为锁维持多个等待队列,比如一个生产者队列,一个消费者队列,一边提高锁的效率。

显式锁(ReentrantLock)正式为了解决这些灵活需求而生。ReentrantLock的字面意思是可重入锁,可重入的意思是线程可以同时多次请求同一把锁,而不会自己导致自己死锁。下面是内置锁和显式锁的区别:

可定时:RenentrantLock.tryLock(long timeout, TimeUnit unit)提供了一种以定时结束等待的方式,如果线程在指定的时间内没有获得锁,该方法就会返回false并结束线程等待。

可中断:你一定见过InterruptedException,很多跟多线程相关的方法会抛出该异常,这个异常并不是一个缺陷导致的负担,而是一种必须,或者说是一件好事。可中断性给我们提供了一种让线程提前结束的方式(而不是非得等到线程执行结束),这对于要取消耗时的任务非常有用。对于内置锁,线程拿不到内置锁就会一直等待,除了获取锁没有其他办法能够让其结束等待。RenentrantLock.lockInterruptibly()给我们提供了一种以中断结束等待的方式。

条件队列(condition queue):线程在获取锁之后,可能会由于等待某个条件发生而进入等待状态(内置锁通过Object.wait()方法,显式锁通过Condition.await()方法),进入等待状态的线程会挂起并自动释放锁,这些线程会被放入到条件队列当中。synchronized对应的只有一个条件队列,而ReentrantLock可以有多个条件队列,多个队列有什么好处呢?请往下看。

条件谓词:线程在获取锁之后,有时候还需要等待某个条件满足才能做事情,比如生产者需要等到“缓存不满”才能往队列里放入消息,而消费者需要等到“缓存非空”才能从队列里取出消息。这些条件被称作条件谓词,线程需要先获取锁,然后判断条件谓词是否满足,如果不满足就不往下执行,相应的线程就会放弃执行权并自动释放锁。使用同一把锁的不同的线程可能有不同的条件谓词,如果只有一个条件队列,当某个条件谓词满足时就无法判断该唤醒条件队列里的哪一个线程;但是如果每个条件谓词都有一个单独的条件队列,当某个条件满足时我们就知道应该唤醒对应队列上的线程(内置锁通过Object.notify()或者Object.notifyAll()方法唤醒,显式锁通过Condition.signal()或者Condition.signalAll()方法唤醒)。这就是多个条件队列的好处。

使用内置锁时,对象本身既是一把锁又是一个条件队列;使用显式锁时,RenentrantLock的对象是锁,条件队列通过RenentrantLock.newCondition()方法获取,多次调用该方法可以得到多个条件队列。

一个使用显式锁的典型示例如下:

// 显式锁的使用示例

ReentrantLock lock = new ReentrantLock();

// 获取锁,这是跟synchronized关键字对应的用法。

lock.lock();

try{

// your code

}finally{

lock.unlock();

}

// 可定时,超过指定时间为得到锁就放弃

try {

lock.tryLock(10, TimeUnit.SECONDS);

try {

// your code

}finally {

lock.unlock();

}

} catch (InterruptedException e1) {

// exception handling

}

// 可中断,等待获取锁的过程中线程线程可被中断

try {

lock.lockInterruptibly();

try {

// your code

}finally {

lock.unlock();

}

} catch (InterruptedException e) {

// exception handling

}

// 多个等待队列,具体参考[ArrayBlockingQueue](https://github.com/CarpenterLee/JCRecipes/blob/master/markdown/ArrayBlockingQueue.md)

/** Condition for waiting takes */

private final Condition notEmpty = lock.newCondition();

/** Condition for waiting puts */

private final Condition notFull = lock.newCondition();

注意,上述代码将unlock()放在finally块里,这么做是必需的。显式锁不像内置锁那样会自动释放,使用显式锁一定要在finally块中手动释放,如果获取锁后由于异常的原因没有释放锁,那么这把锁将永远得不到释放!将unlock()放在finally块中,保证无论发生什么都能够正常释放。

结论

内置锁能够解决大部分需要同步的场景,只有在需要额外灵活性是才需要考虑显式锁,比如可定时、可中断、多等待队列等特性。

显式锁虽然灵活,但是需要显式的申请和释放,并且释放一定要放到finally块中,否则可能会因为异常导致锁永远无法释放!这是显式锁最明显的缺点。

综上,当需要同步时请优先考虑更安全的更易用的隐式锁。

java 内置锁_深入理解java内置锁(synchronized)和显式锁(ReentrantLock)相关推荐

  1. Java虚拟机不能满足_深入理解Java虚拟机--读书笔记1/3

    <深入理解Java虚拟机-JVM高级特性与最佳实践> Chap 2 Java内存区域与内存溢出异常 1.Java运行时数据区域 A.程序计数器:当前线程所执行字节码的行号指示器,线程私有( ...

  2. 深入理解Java虚拟机知乎_深入理解Java虚拟机(类文件结构)

    深入理解Java虚拟机(类文件结构) 欢迎关注微信公众号:BaronTalk,获取更多精彩好文! 之前在阅读 ASM 文档时,对于已编译类的结构.方法描述符.访问标志.ACC_PUBLIC.ACC_P ...

  3. java虚引用作用_深入理解Java中的引用(二)——强软弱虚引用

    深入理解Java中的引用(二)--强软弱虚引用 在上一篇文章中介绍了Java的Reference类,本篇文章介绍他的四个子类:强引用.软引用.弱引用.虚引用. 强引用(StrongReference) ...

  4. java 接口和虚构_深入理解Java的接口和抽象类

    深入理解Java的接口和抽象类 对于面向对象编程来说,抽象是它的一大特征之一.在Java中,可以通过两种形式来体现OOP的抽象:接口和抽象类.这两者有太多相似的地方,又有太多不同的地方.很多人在初学的 ...

  5. java虚拟机编译顺序_深入理解Java虚拟机(程序编译与代码优化)

    文章首发于微信公众号:BaronTalk,欢迎关注! 对于性能和效率的追求一直是程序开发中永恒不变的宗旨,除了我们自己在编码过程中要充分考虑代码的性能和效率,虚拟机在编译阶段也会对代码进行优化.本文就 ...

  6. java调用子系统代码_深入理解JAVA虚拟机-Idea远程执行本地Java代码 - Java 技术驿站-Java 技术驿站...

    今天在看深入理解JAVA虚拟机的9.3节,作者实现了一个远程执行功能.这个功能可以在远程服务器中临时执行一段程序代码,而去不依赖jdk版本,不改变原有服务端程序的部署,不依赖任何第三方库,不入侵原有的 ...

  7. java抽象类的属性_深入理解Java抽象类

    基础部分内容差不多讲解完了,今天开始进入Java提高篇部分,这部分内容会比之前的内容复杂很多,希望大家做好心理准备,看不懂的部分可以多看两遍,仍不理解的部分那一定是我讲的不够生动,记得留言提醒我. 好 ...

  8. java的弱引用_深入理解Java中的弱引用

    不久之前,我面试了一些求职Java高级开发工程师的应聘者.我常常会面试他们说,"你能给我介绍一些Java中得弱引用吗?",如果面试者这样说,"嗯,是不是垃圾回收有关的?& ...

  9. java 中的流_深入理解Java中的流(Stream)

    首先,流是什么? 流是个抽象的概念,是对输入输出设备的抽象,Java程序中,对于数据的输入/输出操作都是以"流"的方式进行.设备可以是文件,网络,内存等. 流具有方向性,至于是输入 ...

最新文章

  1. 用Python分析深圳程序员工资有多高?
  2. 序列比对-BLAST
  3. ESXi安装全过程及基本配置
  4. 自学python的书籍-不可错过的十本Python好书
  5. 无招胜有招之Java进阶JVM(三)内存模型
  6. deactivate_sending在创建新的table entry时的作用
  7. 5.07—018—周二
  8. Ubuntu更改鼠标灵敏度
  9. 安卓编程用什么软件_手机上能安装PLC编程软件吗?为什么?
  10. WEB前端(4)—— CSS经典案例(DIV+CSS布局)
  11. 【愚公系列】2022年09月 微信小程序-电商项目-UI框架的选型
  12. strlen函数题目
  13. Python生成验证码图片及验证用户提交的验证码是否正确
  14. 特征多项式法(characteristic polynomial )求特征值(结合lanczos和householder)(python,数值积分)
  15. Ubuntu下搭建SVN与Apache权限控制
  16. 利用Python实现图片信息隐藏
  17. android获取整体存储空间大小,Android 获取剩余存储空间
  18. 自动部署项目,全靠它了!
  19. IDEA 2020.1官网汉化插件安装
  20. 凡人修仙传之百炼冥门java_修仙缘:细说我与《凡人修仙传》的过往

热门文章

  1. spark-OutOfMemory:GC overhead limit exceeded 解决,timelimitexceeded
  2. android引入开源项目方法,和解决android-support-v4.jar冲突问题
  3. 如何理解if __name__=='__main__'?
  4. MySQL与MySQLI的异同点
  5. MySQL数据库操作步骤---增删改查
  6. python整数池_对Python中小整数对象池和大整数对象池的使用详解
  7. javascript和python的关系_JavaScript是否越来越像Python?
  8. python大数据处理与分析课程目标_《大数据分析与挖掘》课程教学大纲
  9. 机器学习篇—大厂笔试题
  10. python怎么写出正弦图_如何使用python在图像上生成随机正弦条纹?