[Java] [Lock] [Synchronized VS ReentrantLock]
Overview
- java编写多线程程序时,为了保证线程安全,需要对数据进行同步,经常用到的两种同步方式就是synchronized和重入锁ReentrantLock。
相似点
- 都是加锁方式
- 都是阻塞式同步。即若一个线程获得了对象锁,进入了同步块,其他访问该同步块的线程都必须阻塞在同步块外等待,而进行线程阻塞和唤醒的代价是比较高的(os需要在用户态和内核态之间来回切换)。
区别
- synchronized是java语言的关键字,是原生语法层面的互斥,需要jvm实现;而ReentrantLock是JDK1.5之后提供的API层面的互斥锁,需要lock()和unlock()方法配合try/finnally语句块来完成。
Synchronized
- synchronized通过编译,会在同步块前后分别形成menitorenter和monitorexit这两个字节码。
- 在执行monitorenter指令时,首先要尝试获取对象锁。如果这个对象没被锁定,或者当前线程已经拥有了那个对象锁,把锁的计算器加1,相应的mointorexit的时候减1。
- 当计数器间到0时,锁就被释放了。
- 如果对象获取锁失败,那么当前就要阻塞,直到对象锁被另一个线程释放为止。
- synchronized的重要特性:
- 把代码块声明为synchronized会使代码块具有原子性(即一个线程一次只能执行由一个指定lock保护的代码,从而防止多个线程在更新共享状态时互相冲突)和可见性(用来对付内存缓存和编译器优化的各种反常行为)。
- synchronized存在的不足:
- 无法中断一个正在等候获得锁的线程;
- 无法通过轮询来得到锁;
- 同步锁的释放只能在与获得锁所在的堆栈帧相同的堆栈帧中进行。
ReentrantLock
- ReentrantLock是java.util.concurrent包下提供的一套互斥锁,相比synchronized,提供了一些高级功能:
- 等待可中断:持有锁的线程长期不释放时,正在等待的线程可以选择放弃等待。这相对于Synchronized来说可以避免出现死锁的情况;
- 可支持公平锁:公平锁,即多个线程等待同一个锁时,必须按照申请锁的时间顺序获得锁。synchronized锁是非公平锁;
- 锁绑定多个条件:一个ReentrantLock对象可以同时绑定多个对象。
- ReentrantLock的demo usage:
注意锁必须在finnally块中释放,否则当受保护的代码抛出异常时,锁可能永远得不到释放。- public class SynDemo {
- public static void main(String[] args) {
- Runnable t1 = new MyThread();
- new Thread(t1, "t1").start();
- new Thread(t2, "t2").start();
- }
- }
- class MyThread implements Runnable {
- private Lock lock = new ReentrantLock();
- public void run() {
- lock.lock();
- try {
- for (int i = 0; i < 5; i++) {
- System.out.println(Thread.currentThread.getName() + ":" + i);
- }
- } finally {
- lock.unlock();
- }
- }
- }
- public class SynDemo {
- 因为ReentrantLock是lock的一个抽象,是一个Java类,而不是语言的特性。这就为Lock的多种实现留下了空间,各种实现可能有不同的调度算法、性能特性或者锁定语义。
- ReentrantLock拥有与synchronized相同的并发性和内存语义,但是添加了类似轮询锁、定时锁等候和可中断等候的一些特性。此外,它还提供了在激烈争用情况下的更佳性能。
- reentrant意味着什么呢?简单来说,它有一个与锁相关的获取计数器,如果拥有锁的某个线程再次得到锁,那么获取计数器就加1,然后锁需要被释放两次才能获得真正释放。这模仿了
synchronized
的语义;如果线程进入由线程已经拥有的监控器保护的 synchronized 块,就允许线程继续进行,当线程退出第二个(或者后续)synchronized
块的时候,不释放锁,只有线程退出它进入的监控器保护的第一个synchronized
块时,才释放锁。 - 这些也不意味着reentrantLock可以完全取代synchronized:使用sychronized时不可能忘记释放锁;当JVM用 synchronized 管理锁定请求和释放时,JVM 在生成线程转储时能够包括锁定信息,这些对调试非常有价值,因为它们能标识死锁或者其他异常行为的来源。相比之下,
Lock
类只是普通的类,JVM 不知道具体哪个线程拥有Lock
对象。 - 什么时候用ReentrantLock代替synchronized:在需要ReentrantLock独有的特性(时间锁等候、可中断锁等候、无块结构锁、多个条件变量或者轮询锁)或者是在高度争用的情况下(ReentrantLock具有可伸缩性)使用。
公平锁
- 公平是好事,但是保证公平需要很大的性能成本。
- 要确保公平所需要的记账(bookkeeping)和同步,就意味着被争夺的公平锁要比不公平锁的吞吐率更低。
- synchronized永远是不公平的。但JVM保证了所有线程最终都会得到它们所等候的锁,这意味着统计上的公平。
[Java] [Lock] [Synchronized VS ReentrantLock]的更多相关文章
- Java并发——synchronized和ReentrantLock的联系与区别
0 前言 本文通过使用synchronized以及Lock分别完成"生产消费场景",再引出两种锁机制的关系和区别,以及一些关于锁的知识点. 本文原创,转载请注明出处:http:// ...
- 高级java必会系列二:多线程经常使用的3个关键字:synchronized、ReentrantLock、volatile
系列一讲解了多线程,本章讲解多线程开发中经常使用到的3个关键字synchronized.ReentrantLock.volatile. 一.synchronized 互斥锁,即操作互斥,并发线程过来, ...
- Java Lock ReentrantLock ReentrantReadWriteLock
Lock与Synchronized的区别: 1)Lock是一个接口,而synchronized是Java中的关键字,synchronized是内置的语言实现: 2)synchronized在发生异 ...
- Java:使用synchronized和Lock对象获取对象锁
在并发环境下,解决共享资源冲突问题时,可以考虑使用锁机制. 1.对象的锁 所有对象都自动含有单一的锁. JVM负责跟踪对象被加锁的次数.如果一个对象被解锁,其计数变为0.在任务(线程)第一次给对象加锁 ...
- Java多线程的~~~Lock接口和ReentrantLock使用
在多线程开发.除了synchronized这个keyword外,我们还通过Lock接口来实现这样的效果.由Lock接口来实现 这样的多线程加锁效果的优点是非常的灵活,我们不在须要对整个函数加锁,并且能 ...
- 【Java】synchronized与lock的区别
从Java 5之后,在java.util.concurrent.locks包下提供了另外一种方式来实现同步访问,那就是Lock. 也许有朋友会问,既然都可以通过synchronized来实现同步访问了 ...
- java 多线程 synchronized与lock的通信机制等问题,结合相应实例说明
1. 利用多线程实现如下需求: 写两个线程,一个线程打印1~52,另一个线程打印A~Z,打印顺序是12A34B...5152Z: 2. 使用synchronized 实现 public class T ...
- java的两种同步方式, Synchronized与ReentrantLock的区别
java在编写多线程程序时,为了保证线程安全,需要对数据同步,经常用到两种同步方式就是Synchronized和重入锁ReentrantLock. 相似点: 这两种同步方式有很多相似之处,它们都是加锁 ...
- java多线程 21 : ReentrantReadWriteLock ,synchronized和ReentrantLock的对比
读写锁ReentrantReadWriteLock概述 大型网站中很重要的一块内容就是数据的读写,ReentrantLock虽然具有完全互斥排他的效果(即同一时间只有一个线程正在执行lock后面的任务 ...
随机推荐
- FTR-B3GA003Z-信号继电器
IC名称:B3GA003Z 数据手册:链接:https://pan.baidu.com/s/1MNe-fcKNAnuXyRLxhfUhjA 密码:6mo2 芯片自拍照: 封装尺寸以及引脚说明: 信号电 ...
- Grunt: 拼接代码,js丑化(压缩),css压缩,html压缩,观察文件,拷贝文件,删除文件,压缩文件
准备工作 grunt 基于nodeJs所以 nodeJs需要的基础配置都需要安装 1.Grunt 安装 npm install -g grunt-cli 这是全局安装 2.在当前文件下npm init ...
- mysql 存储session
https://www.cnblogs.com/cndavidwang/p/3930619.html
- Html p 标签
Html p 标签 <html> <body> <!-- p标签:输出跳到下一段落--> <p>内容</p> 注:段落之前各空出一行. &l ...
- kmp匹配详解
字符串算法都是毒瘤的 一.kmp算法的用处 在文本串中查找模式串的位置,数量 文本串:要在这个字符串查找模式串 模式串:在文本串中查找的字符串 全是废话 二.kmp算法的思想 话说kmp好像是3个发明 ...
- PHP遍历目录和文件及子目录和文件
正常直接使用opendir方法,就可以读到所有的目录和文件 文件可以直接记录下来,目录则需要再进一步获取里边的文件信息 也就是,如果当前读出来是目录,则需要再次调用函数本身(递归),直到没有目录 循环 ...
- time&datetime模块详解
一.time模块 1.时间格式转换图: 2.time模块中时间表现的格式主要有三种: a.timestamp时间戳,时间戳表示的是从1970年1月1日00:00:00开始按秒计算的偏移量 b.for ...
- Codeforces 17E Palisection - Manacher
题目传送门 传送点I 传送点II 传送点III 题目大意 给定一个串$s$询问,有多少对回文子串有交. 好像很简单的样子. 考虑能不能直接求,感觉有点麻烦.因为要考虑右端点在当前回文子串内还有区间包含 ...
- maven 执行testng.xml文件失败解决问题
在pom.xml中配置了testng的依赖后,在surefire-plugin中又配置了suitexmlfiles指向testng.xml文件,但是使用mvn test运行时,没有运行testng.x ...
- es6 class的基本语法
ES5以及之前的版本,没有类的概念,但是聪明的JavaScript开发者,为了实现面向对象,创建了特殊的近类结构. ES5中创建类的方法:新建一个构造函数,定义一个方法并且赋值给构造函数的原型. 'u ...