一、使用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

Lock类可以完全替代synchronize,并且具有synchronize没有的其他特性。

六、公平锁与非公平锁

锁Lock分为公平锁和非公平锁,

  • 公平锁:线程获取锁的顺序是按照线程加锁的顺序来分配的,即先进先出;
  • 非公平锁:获取锁的抢占机制,非先进先出,可能造成某些线程一直拿不到锁;

如何实现公平和非公平?

  • ReentrantLock有一个构造函数,可以接收一个boolean参数,来决定是公平锁还是非公平锁。

END

《JAVA多线程编程核心技术》 笔记:第四章、Lock的使用的更多相关文章

  1. Java多线程编程核心技术,第四章

    1,ReentrantLock 2,object的wait(),wait(x),notify(),notifyAll(),分别等于Condition的await(),await(x,y),signal ...

  2. java多线程编程核心技术-笔记

    一.第一章 1.自定义线程类中实例变量针对其他线程有共享和不共享之分,自定义线程中的变量,如果是继承自thread类,则每个线程中的示例变量的更改,不影响其他线程2.当多个线程去访问一个局部变量是会产 ...

  3. Java多线程编程核心技术,第六章

    1,饿汉模式/单例模式,一开始就新建一个静态变量,后面用getInstance()都是同一个变量 2,懒汉模式/单例模式,在getInstance()才会new一个对象,在第一个有了后不会继续创建 3 ...

  4. Java多线程编程核心技术,第五章

    1,Timer timer = new Timer(true)现在是守护进程 2,timer是按照顺的,没有异步 3,timer方法,schedule(TimerTask task, Date fir ...

  5. Java多线程编程核心技术,第三章

    1,notify的同步块完了,才会运行wait的同步块 2,interrupt()不是静态方法,用在wait的线程上会有InteruptException,锁也会被释放 3,notify()唤醒的线程 ...

  6. 《Java 多线程编程核心技术》- 笔记

    作为业务开发人员,能够在工作中用到的技术其实不多.虽然平时老是说什么,多线程,并发,注入,攻击!但是在实际工作中,这些东西不见得用得上.因为,我们用的框架已经把这些事做掉了. 比如web开发,外面有大 ...

  7. java并发编程的艺术——第四章总结

    第四章并发编程基础 4.1线程简介 4.2启动与终止线程 4.3线程间通信 4.4线程应用实例 java语言是内置对多线程支持的. 为什么使用多线程: 首先线程是操作系统最小的调度单元,多核心.多个线 ...

  8. Java多线程编程核心技术(三)多线程通信

    线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一,可以说,使线程间进行通信后,系统之间的交互性会更强大,在大大提高CPU利用率的同时 ...

  9. Java多线程编程核心技术

    Java多线程编程核心技术 这本书有利于对Java多线程API的理解,但不容易从中总结规律. JDK文档 1. Thread类 部分源码: public class Thread implements ...

  10. 《Java多线程编程核心技术》推荐

    写这篇博客主要是给猿友们推荐一本书<Java多线程编程核心技术>. 之所以要推荐它,主要因为这本书写得十分通俗易懂,以实例贯穿整本书,使得原本抽象的概念,理解起来不再抽象. 只要你有一点点 ...

随机推荐

  1. 设备管理器里“SM总线控制器”、“其它PCI桥设备”驱动有问题

    WinXP重装系统后设备管理器里面出现黄色问号.各自是"SM总线控制器"和"其它PCI桥设备",主板是七彩虹的,芯片组是 geForce 7025的,南桥是 n ...

  2. 开始使用Bootstrap

    bootstrap使用到的图标字体文件格式有 .woff,IIS7下需要添加MIME映射:.woff  application/x-font-woff

  3. Windows 7/8/8.1 误删EFI启动项,无法开机解决方式(U盘+原版镜像)

    今天手贱把Windows 7的启动项删了.由于是GPT分区,EFI引导的,又不像MBR那般easy解决 想想重装系统也麻烦,并且仅仅是删了个启动项而已.就不是必需去费那个时间 想了一下,Windows ...

  4. winfrom更新

    原理: 工具生成更新配置节xml放到文件服务器上,外网可访问: 能过本地配置文件与服务器配置文件日期属性对比及配置节版本与大小属性判断有无更新: 存在更新,将文件从服务器下载到客户端,并替换原程序重启 ...

  5. MySQL加速查询速度的独门武器:查询缓存

    [导读] 与朋友或同事谈到MySQL查询缓存功能的时候,个人喜欢把查询缓存功能Query Cache比作荔枝, 是非常营养的东西,但是一次性吃太多了,就容易上火而流鼻血,虽然不是特别恰当的比喻,但是有 ...

  6. 测试-一个unity的编译bug,初始化器

    .net C#下测试: public class Class1 { public bool toggle1 = true; public bool toggle2; } 一个结构类Class1,对里面 ...

  7. bootstrip可视化布局

    http://www.w3cschool.cc/try/bootstrap/layoutit/

  8. 自定义textView的placeholder和边框

    想实现的效果: // //  LHQsuggestionViewCtrl.m //  A13 - 设置 // //  Created by vic fan on 16/6/23. //  Copyri ...

  9. curl编译

    首先下载并解压: wget http://curl.haxx.se/download/curl-7.37.1.tar.gz tar -xzvf curl-7.37.1.tar.gz 编译安装: cd ...

  10. VMware网络连接失败解决方法

    假如你碰到了VMware 网络被断开,明明已经分配了适配器,客户端却显示网络断开没有连接. 一. 可用恢复默认的方法重置所有网卡及服务. 如图片操作: 进主工具首页.点击: 虚拟网络编辑器 然后点击下 ...