AQS为在独占模式下获取锁分别提供三种获取方式:
- 不响应线程中断获取;
- 响应线程中断获取;
- 设置超时时间获取。
这三种方式整体步骤大致是相同的,只有少部分不同的地方:
第一种在获取时会忽略中断;
而第二种则是获取时响应中断;
第三种是获取时,如果超时则立即返回。
不响应线程中断获取锁
1 | //不响应中断方式获取(独占模式) |
acquire方法是获取锁的基础,这个方法会忽略中断,意思是说如果节点对应的线程中断,则acquire()方法会忽略,只有从同步队列中返回true才最终调用selfInterrupt方法响应中断。代码很简单,但是它按照顺序执行了下图所示的4个步骤。
第一步:!tryAcquire(arg)
尝试获取资源state
1 | //尝试获取锁(独占模式) |
这时候来了一个人,他首先尝试着去敲了敲门,如果发现门没锁(tryAcquire(arg)=true),那就直接进去了。如果发现门锁了(tryAcquire(arg)=false),就执行下一步。这个tryAcquire方法决定了什么时候锁是开着的,什么时候锁是关闭的。这个方法必须要让子类去覆盖,重写里面的判断逻辑。
第二步:addWaiter(Node.EXCLUSIVE)
获取资源失败,则封装成Node结点加入到同步队列中。
1 | //将当前线程包装成结点并添加到同步队列尾部 |
执行到这一步表明第一次获取锁失败,那么这个人就给自己领了块号码牌进入排队区去排队了,在领号码牌的时候会声明自己想要以什么样的方式来占用房间(独占模式or共享模式)。注意,这时候他并没有坐下来休息(将自己挂起)。
第三步:acquireQueued(addWaiter(Node.EXCLUSIVE), arg)
在同步队列中获取资源。其实这个方法非常的好理解,前面已经尝试获取资源,但是失败了,并且加到了同步队列中,在等待中可以判断自己是否可以休息以下,如果可以休息,那就等待着其他线程唤醒自己,在继续获取资源,直到成功才返回。
1 | //以不可中断方式获取锁(独占模式) |
领完号码牌进入排队区后就会立马执行这个方法,当一个结点首次进入排队区后有两种情况,一种是发现他前面的那个人已经离开座位进入房间了,那他就不坐下来休息了,会再次去敲一敲门看看那小子有没有完事。如果里面的人刚好完事出来了,都不用他叫自己就直接冲进去了。否则,就要考虑坐下来休息一会儿了,但是他还是不放心,如果他坐下来睡着后没人提醒他怎么办?他就在前面那人的座位上留一个小纸条,好让从里面出来的人看到纸条后能够唤醒他。还有一种情况是,当他进入排队区后发现前面还有好几个人在座位上排队呢,那他就可以安心的坐下来咪一会儿了,但在此之前他还是会在前面那人(此时已经睡着了)的座位上留一个纸条,好让这个人在走之前能够去唤醒自己。当一切事情办妥了之后,他就安安心心的睡觉了,注意,我们看到整个for循环就只有一个出口,那就是等线程成功的获取到锁之后才能出去,在没有获取到锁之前就一直是挂在for循环的parkAndCheckInterrupt()方法里头。线程被唤醒后也是从这个地方继续执行for循环。
第四步:selfInterrupt()
1 | //当前线程将自己中断 |
由于上面整个线程一直是挂在for循环的parkAndCheckInterrupt()方法里头,没有成功获取到锁之前不响应任何形式的线程中断,只有当线程成功获取到锁并从for循环出来后,他才会查看在这期间是否有人要求中断线程,如果是的话再去调用selfInterrupt()方法将自己挂起。
响应线程中断获取锁
1 | public final void acquireInterruptibly(int arg) throws InterruptedException { |
1 | //以可中断模式获取锁(独占模式) |
响应线程中断方式和不响应线程中断方式获取锁流程上大致上是相同的。唯一的一点区别就是线程从parkAndCheckInterrupt方法中醒来后会检查线程是否中断,如果是的话就抛出InterruptedException异常,而不响应线程中断获取锁是在收到中断请求后只是设置一下中断状态,并不会立马结束当前获取锁的方法,一直到结点成功获取到锁之后才会根据中断状态决定是否将自己挂起。
设置超时时间获取锁
1 | public final boolean tryAcquireNanos(int arg, long nanosTimeout) throws InterruptedException { |
1 | //以限定超时时间获取锁(独占模式) |
设置超时时间获取首先会去获取一下锁,第一次获取锁失败后会根据情况,如果传入的超时时间大于自旋时间那么就会将线程挂起一段时间,否则的话就会进行自旋,每次获取锁之后都会将超时时间减去获取一次锁所用的时间。一直到超时时间小于0也就说明超时时间用完了,那么这时就会结束获取锁的操作然后返回获取失败标志。注意在以超时时间获取锁的过程中是可以响应线程中断请求的。
独占模式下释放锁
1 | //释放锁的操作(独占模式) |
线程持有锁进入房间后就会去办自己的事情,等事情办完后它就会释放锁并离开房间。通过tryRelease方法可以拨动密码锁进行解锁,tryRelease方法是需要让子类去覆盖的,不同的子类实现的规则不一样,也就是说不同的子类设置的密码不一样。
像在ReentrantLock当中,房间里面的人每调用tryRelease方法一次,state就减1,直到state减到0的时候密码锁就开了。这个过程就像在不停的转动密码锁的转轮,而每次转动转轮数字只是减少1。
CountDownLatch和这个也有点类似,只不过它不是一个人在转,而是多个人每人都去转一下,集中大家的力量把锁给开了。
线程出了房间后它会找到自己原先的座位,也就是找到head结点。看看座位上有没有人给它留了小纸条,如果有的话它就知道有人睡着了需要让它帮忙唤醒,那么它就会去唤醒那个线程。如果没有的话就表明同步队列中暂时还没有人在等待,也没有人需要它唤醒,所以它就可以安心的离去了。