1 Overview java
若是查看ReentrantLock,CountDownLatch,Semaphore,FutureTask,ThreadPoolExecutor的源码,都会发现有个名叫Sync的静态内部类,继承自AbstractQueuedSynchronizer。实际上AbstractQueuedSynchronizer是java.util.concurrent的核心组件之一,它为并发包中的其余synchronizers提供了一组公共的基础设施。 并发
2 LockSupport
在介绍AbstractQueuedSynchronizer以前,首先要介绍一下java.util.concurrent.locks.LockSupport。在LockSupport出现以前,若是要block/unblock某个Thread,除了使用Java语言内置的monitor机制以外,只能经过Thread.suspend()和Thread.resume()。然而Thread.suspend()和Thread.resume()基本上不可用,除了可能致使死锁以外,它们还存在一个没法解决的竞争条件:若是在调用Thread.suspend()以前调用了Thread.resume(),那么该Thread.resume()调用没有任何效果。LockSupport最主要的做用,即是经过一个许可(permit)状态,解决了这个问题。 工具
那么LockSupport和Java语言内置的monitor机制有什么区别呢?它们的语义是不一样的。LockSupport是针对特定Thread来进行block/unblock操做的;wait()/notify()/notifyAll()是用来操做特定对象的等待集合的。为了防止知识生锈,在这里简单介绍一下Java语言内置的monitor机制(详见:http://whitesock.iteye.com/blog/162344 )。正如每一个Object都有一个锁, 每一个Object也有一个等待集合(wait set),它有wait、notify、notifyAll和Thread.interrupt方法来操做。同时拥有锁和等待集合的实体,一般被成为监视器(monitor)。每一个Object的等待集合是由JVM维护的。等待集合一直存放着那些由于调用对象的wait方法而被阻塞的线程。因为等待集合和锁之间的交互机制,只有得到目标对象的同步锁时,才能够调用它的wait、notify和notifyAll方法。这种要求一般没法靠编译来检查,若是条件不能知足,那么在运行的时候调用以上方法就会致使其抛出IllegalMonitorStateException。 spa
wait() 方法被调用后,会执行以下操做: .net
notify()方法被调用后,会执行以下操做: 线程
notifyAll()方法被调用后的操做和notify()相似,不一样的只是等待集合中全部的线程(同时)都要执行那些操做。然而等待集合中的线程必需要在竞争到目标对象的同步锁以后,才能继续执行。 对象
LockSupport类中比较重要的方法有以下几个: blog
其中park()和park(Object blocker)方法用于block当前线程,unpark(Thread thread)方法用于unblock制定的线程。 跟Thread.suspend()和Thread.resume()不一样的是,LockSupport经过许可(permit)机制保证:若是当前线程拥有许可,那么park系列方法会消费掉该许可,而且当即返回(不会被阻塞)。也就是说以下代码在执行的时候,不会被阻塞: 继承
须要注意的是:许可不会被累计。也就是说在park调用以前的屡次unpark调用,只会unblock一次park调用。即如下代码会被阻塞: 文档
关于park()和park(Object blocker)的区别,Object blocker参数的做用在于容许记录当前线程被阻塞的缘由,以便监控分析工具进行分析。官方的文档中也更建议使用park(Object blocker)。此外,跟Object.wait()方法同样,park系列方法也会由于伪唤醒的缘由返回。