《JAVA多线程编程核心技术》 笔记:第四章、Lock的使用
1.1 ReentrantLock的使用:
1.2 ReentrantLock的不足:
1.3 正确使用Condition实现等待/通知
1.4 使用多个Condition实现通知部分线程:正确用法
1.5 各种方法的测试
二、使用ReentrantReadWriteLock类
2.1 类ReentrantReadWriteLock初识
三、对比:ReentrantLock和ReentrantReadWriteLock
五、Lock类和synchronize
六、公平锁与非公平锁
END
一、使用ReentrantLock类
1.1 ReentrantLock的使用:
和synchronized没太多区别,只是要自己lock和unlock
1.2 ReentrantLock的不足:
ReentrantLock具有完全互斥排他的效果,即同一时间只有一个线程在执行ReentrantLock.lock()方法后面的任务。这样保证了实例变量的线程安全性,但是效率非常低下。
所以JDK中提供了读写锁ReentrantReadWriteLock,可以加快运行效率。
1.3 正确使用Condition实现等待/通知
调用condition.await()之前,必须调用lock.lock();获取到对应的锁对象。
相同
| Object类 | 相当于 | Condition类 |
|---|---|---|
| wait()方法 | = | await()方法 |
| wait(long)方法 | = | await(long time, TimeUnit unit)方法 |
| notify()方法 | = | signal()方法 |
| notifyAll()方法 | = | signalAll()方法 |
区别:
Condition可以实现选择通知;
而notify(),通知的对象随意,notifyAll()则通知所有对象。
synchronized相当于整个Lock对象中只有一个单一的Condition对象。
1.4 使用多个Condition实现通知部分线程:正确用法
如果使Condition对象可以唤醒部分指定线程,需先对线程进行分组,再唤醒指定组中的线程。
实现的方式是:类中有两个Condition对象。不同线程在不同的Condition对象上await(),然后调用Condition的signalAll()即可
1.5 各种方法的测试
| 方法 | 解释 |
|---|---|
| getHoldCount() | 查询当前线程保持锁定的个数,也就是调用lock()方法的次数; |
| getQueueLength() | 返回正等待获取此锁定的线程估计数;即等待该锁释放,然后获取锁的线程数 |
| getWaitQueueLength() | 返回等待与此锁定相关的给定条件Condition的线程估计数,即已经在该锁上执行了await()方法,并且没有被signal唤醒的线程数; |
| 方法 | 解释 |
|---|---|
| hasQueuedThread() | 查询指定的线程是否正在等待获取此锁定。 |
| hasQueuedThreads() | 查询是否有线程正在等待获取此锁定。 |
| hasWaiters() | 是否有线程正在等待与此锁定有关的condition条件; |
| 方法 | 解释 |
|---|---|
| isFair() | 判断是不是公平锁 |
| isHeldByCurrentThread() | 查询当前线程是否保持此锁定 |
| isLocked() | 查询此锁定是否由任意线程保持; |
| 方法 | 解释 |
|---|---|
| lockInterruptibly() | 如果当前线程未被中断,则获取锁定,如果已被中断则出现异常。 |
| tryLock() | 仅在:调用时该锁定未被另一个线程保持的情况下,才获取该锁定。 |
| tryLock(long timeout,TimeUnit unit) | 如果该锁定在给定等待时间内没有被另一个线程保持,且当前线程未被中断,则获取该锁定。(如果当前该锁没有被另一线程保持,那么直接获取该锁;如果当前该锁已被另一线程保持,则等待对应时间,然后再获取该锁。能否获取到结合具体场景分析) |
| 方法 | 解释 |
|---|---|
| awaitUninterruptibly() | 不是很理解..即使被中断也什么都不做?不抛异常? |
| awaitUntil(Date date) | 等待到达这个时间点。如果在等待时没有被其他线程唤醒,则在时间点达到会自动被唤醒;如果在等待时被其他线程唤醒,那么这个时间就没有太多意义了。 |
二、使用ReentrantReadWriteLock类
2.1 类ReentrantReadWriteLock初识
ReentrantLock具有完全互斥排他的效果,即同一时间只有一个线程在执行ReentrantLock.lock()方法后面的任务。这样保证了实例变量的线程安全性,但是效率非常低下。
所以JDK中提供了读写锁ReentrantReadWriteLock,可以加快运行效率。
ReentrantReadWriteLock有两个锁
- 读锁:读操作相关,也称为共享锁;
- 写锁:写操作相关,也称为排他锁;
多个读锁之间不互斥,读锁写锁互斥,写锁写锁互斥
即多个线程可以同时读取,但是同一时刻只允许一个线程写入。
三、对比:ReentrantLock和ReentrantReadWriteLock
ReentrantLock
ReentrantReadWriteLock
锁种类
只有一种锁
互斥情况
完全互斥
理解
ReentrantLock只相当于
ReentrantReadWriteLock的写锁
五、Lock类和synchronize
| ReentrantLock | ReentrantReadWriteLock |
|---|---|
| 锁种类 | 只有一种锁 |
| 互斥情况 | 完全互斥 |
| 理解 | ReentrantLock只相当于 ReentrantReadWriteLock的写锁 |
Lock类可以完全替代synchronize,并且具有synchronize没有的其他特性。
六、公平锁与非公平锁
锁Lock分为公平锁和非公平锁,
- 公平锁:线程获取锁的顺序是按照线程加锁的顺序来分配的,即先进先出;
- 非公平锁:获取锁的抢占机制,非先进先出,可能造成某些线程一直拿不到锁;
如何实现公平和非公平?
- ReentrantLock有一个构造函数,可以接收一个boolean参数,来决定是公平锁还是非公平锁。
END
《JAVA多线程编程核心技术》 笔记:第四章、Lock的使用的更多相关文章
- Java多线程编程核心技术,第四章
1,ReentrantLock 2,object的wait(),wait(x),notify(),notifyAll(),分别等于Condition的await(),await(x,y),signal ...
- java多线程编程核心技术-笔记
一.第一章 1.自定义线程类中实例变量针对其他线程有共享和不共享之分,自定义线程中的变量,如果是继承自thread类,则每个线程中的示例变量的更改,不影响其他线程2.当多个线程去访问一个局部变量是会产 ...
- Java多线程编程核心技术,第六章
1,饿汉模式/单例模式,一开始就新建一个静态变量,后面用getInstance()都是同一个变量 2,懒汉模式/单例模式,在getInstance()才会new一个对象,在第一个有了后不会继续创建 3 ...
- Java多线程编程核心技术,第五章
1,Timer timer = new Timer(true)现在是守护进程 2,timer是按照顺的,没有异步 3,timer方法,schedule(TimerTask task, Date fir ...
- Java多线程编程核心技术,第三章
1,notify的同步块完了,才会运行wait的同步块 2,interrupt()不是静态方法,用在wait的线程上会有InteruptException,锁也会被释放 3,notify()唤醒的线程 ...
- 《Java 多线程编程核心技术》- 笔记
作为业务开发人员,能够在工作中用到的技术其实不多.虽然平时老是说什么,多线程,并发,注入,攻击!但是在实际工作中,这些东西不见得用得上.因为,我们用的框架已经把这些事做掉了. 比如web开发,外面有大 ...
- java并发编程的艺术——第四章总结
第四章并发编程基础 4.1线程简介 4.2启动与终止线程 4.3线程间通信 4.4线程应用实例 java语言是内置对多线程支持的. 为什么使用多线程: 首先线程是操作系统最小的调度单元,多核心.多个线 ...
- Java多线程编程核心技术(三)多线程通信
线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一,可以说,使线程间进行通信后,系统之间的交互性会更强大,在大大提高CPU利用率的同时 ...
- Java多线程编程核心技术
Java多线程编程核心技术 这本书有利于对Java多线程API的理解,但不容易从中总结规律. JDK文档 1. Thread类 部分源码: public class Thread implements ...
- 《Java多线程编程核心技术》推荐
写这篇博客主要是给猿友们推荐一本书<Java多线程编程核心技术>. 之所以要推荐它,主要因为这本书写得十分通俗易懂,以实例贯穿整本书,使得原本抽象的概念,理解起来不再抽象. 只要你有一点点 ...
随机推荐
- C#指南,重温基础,展望远方!(3)类型和变量
C# 有两种类型:值类型和引用类型. 值类型的变量直接包含数据,而引用类型的变量则存储对数据(称为“对象”)的引用.对于引用类型,两个变量可以引用同一对象:因此,对一个变量执行的运算可能会影响另一个变 ...
- andorid手机电脑操作
之前一直使用androidscreencast在pc上对手机进行操作,好久都没用了,前些天再次用的时候,提演示样例如以下: 决定还是自己写一个吧,由于7月份要做一个小分享,打算讲一些android的东 ...
- DataURL与File,Blob,canvas对象之间的互相转换的Javascript (未完)
canvas转换为dataURL (从canvas获取dataURL) var dataurl = canvas.toDataURL('image/png'); var dataurl2 = canv ...
- Sublime Text 编辑器 插件 之 "Sublime Alignment" 详解
作者:shede333主页:http://my.oschina.net/shede333版权声明:原创文章,版权声明:自由转载-非商用-非衍生-保持署名 | [Creative Commons BY- ...
- plist 与 JOSN的解析
- [转]ubuntu安装gcc
Ubuntu缺省情况下,并没有提供C/C++的编译环境,因此还需要手动安装. 如果单独安装gcc以及g++比较麻烦,幸运的是,为了能够编译Ubuntu的内核,Ubuntu提供了一个build-esse ...
- 替换jar中的指定文件
替换jar 包下面的class 文件,很多人会想到直接用winrar 打开替换,在一般的情况下, 是可行的,但是如果说这个jar 的代码经过混淆后,会有大小写不同,文件名是相同的,在windos ...
- python 差分包制作-如何来制作差分包?
继百度网盘爬虫,百度图片爬虫后这是本人第三篇有关python的文章了,由于本人之前做过嵌入式的一些东西,其中会涉及到差分包的制作,所以这篇文章想谈谈如何利用python来制作差分包,如果你对嵌入式的东 ...
- linux下安装python的第三方module
1.首先需要有python环境 2.安装pip软件:下载地址,https://pypi.python.org/pypi/pip/6.0.8 解压pip的压缩包:sudo tar -zxvf pip-6 ...
- Ubuntu Apache配置及开启mod_rewrite模块
刚刚将服务器系统从CentOS换成Ubuntu,将MySQL,Apache,PHP和Wordpress安装好后,发现打开主页是正常的,但是打开文章页面时出现错误.因为使用了自定义的固定链接设置,那自然 ...
1,ReentrantLock 2,object的wait(),wait(x),notify(),notifyAll(),分别等于Condition的await(),await(x,y),signal ...
一.第一章 1.自定义线程类中实例变量针对其他线程有共享和不共享之分,自定义线程中的变量,如果是继承自thread类,则每个线程中的示例变量的更改,不影响其他线程2.当多个线程去访问一个局部变量是会产 ...
1,饿汉模式/单例模式,一开始就新建一个静态变量,后面用getInstance()都是同一个变量 2,懒汉模式/单例模式,在getInstance()才会new一个对象,在第一个有了后不会继续创建 3 ...
1,Timer timer = new Timer(true)现在是守护进程 2,timer是按照顺的,没有异步 3,timer方法,schedule(TimerTask task, Date fir ...
1,notify的同步块完了,才会运行wait的同步块 2,interrupt()不是静态方法,用在wait的线程上会有InteruptException,锁也会被释放 3,notify()唤醒的线程 ...
作为业务开发人员,能够在工作中用到的技术其实不多.虽然平时老是说什么,多线程,并发,注入,攻击!但是在实际工作中,这些东西不见得用得上.因为,我们用的框架已经把这些事做掉了. 比如web开发,外面有大 ...
第四章并发编程基础 4.1线程简介 4.2启动与终止线程 4.3线程间通信 4.4线程应用实例 java语言是内置对多线程支持的. 为什么使用多线程: 首先线程是操作系统最小的调度单元,多核心.多个线 ...
线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一,可以说,使线程间进行通信后,系统之间的交互性会更强大,在大大提高CPU利用率的同时 ...
Java多线程编程核心技术 这本书有利于对Java多线程API的理解,但不容易从中总结规律. JDK文档 1. Thread类 部分源码: public class Thread implements ...
写这篇博客主要是给猿友们推荐一本书<Java多线程编程核心技术>. 之所以要推荐它,主要因为这本书写得十分通俗易懂,以实例贯穿整本书,使得原本抽象的概念,理解起来不再抽象. 只要你有一点点 ...
C# 有两种类型:值类型和引用类型. 值类型的变量直接包含数据,而引用类型的变量则存储对数据(称为“对象”)的引用.对于引用类型,两个变量可以引用同一对象:因此,对一个变量执行的运算可能会影响另一个变 ...
之前一直使用androidscreencast在pc上对手机进行操作,好久都没用了,前些天再次用的时候,提演示样例如以下: 决定还是自己写一个吧,由于7月份要做一个小分享,打算讲一些android的东 ...
canvas转换为dataURL (从canvas获取dataURL) var dataurl = canvas.toDataURL('image/png'); var dataurl2 = canv ...
作者:shede333主页:http://my.oschina.net/shede333版权声明:原创文章,版权声明:自由转载-非商用-非衍生-保持署名 | [Creative Commons BY- ...
Ubuntu缺省情况下,并没有提供C/C++的编译环境,因此还需要手动安装. 如果单独安装gcc以及g++比较麻烦,幸运的是,为了能够编译Ubuntu的内核,Ubuntu提供了一个build-esse ...
替换jar 包下面的class 文件,很多人会想到直接用winrar 打开替换,在一般的情况下, 是可行的,但是如果说这个jar 的代码经过混淆后,会有大小写不同,文件名是相同的,在windos ...
继百度网盘爬虫,百度图片爬虫后这是本人第三篇有关python的文章了,由于本人之前做过嵌入式的一些东西,其中会涉及到差分包的制作,所以这篇文章想谈谈如何利用python来制作差分包,如果你对嵌入式的东 ...
1.首先需要有python环境 2.安装pip软件:下载地址,https://pypi.python.org/pypi/pip/6.0.8 解压pip的压缩包:sudo tar -zxvf pip-6 ...
刚刚将服务器系统从CentOS换成Ubuntu,将MySQL,Apache,PHP和Wordpress安装好后,发现打开主页是正常的,但是打开文章页面时出现错误.因为使用了自定义的固定链接设置,那自然 ...