本文主要是介绍Java基础知识总结(79),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
waitStatus属性
每个节点与等待线程关联,每个节点维护一个状态waitStatus,waitStatus的各种值以常量的形式进行定义。
-
CANCELLED(1):waitStatus值为1时表示该线程节点已释放(超时、中断),已取消的节点不会再阻塞,表示线程因为中断或者等待超时,需要从等待队列中取消等待。由于该节点线程等待超时或者被中断,需要从同步队列中取消等待,因此该线程被置1。节点进入了取消状态,该类型节点不会参与竞争,且会一直保持取消状态。
-
SIGNAL(-1):waitStatus为SIGNAL(-1)时表示其后继的节点处于等待状态,当前节点对应的线程如果释放了同步状态或者被取消,就会通知后继节点,使后继节点的线程得以运行。
-
CONDITION(-2):waitStatus为-2时,表示该线程在条件队列中阻塞(主要在Condition中使用),表示节点在等待队列中,当持有锁的线程调用了CONDITION的signal()方法之后,节点会从该CONDITION的等待队列转移到该锁的同步队列上,去竞争锁(注意:这里的同步队列就是我们讲的AQS维护的FIFO队列,等待队列则是每个CONDITION关联的队列)。
-
PROPAGATE(-3):waitStatus为-3时,表示下一个线程获取共享锁后,自己的共享状态会被无条件地传播下去,因为共享锁可能出现同时有N个锁可以用,这时直接让后面的N个节点都来工作。共享锁是可以多个线程共有的,当一个节点的线程获取共享锁后,必然要通知后继共享节点的线程也可以获取锁了,这样就不会让其他等待的线程等很久,这种向后通知(传播)的目的也是尽快通知其他等待的线程尽快获取锁。
-
0:waitStatus为0时,表示当前节点处于初始状态。
thread
Node的thread成员用来存放进入AQS队列中的线程引用,Node的nextWaiter成员用来指向自己的后继等待节点,此成员只有线程处于条件等待队列中的时候使用。
CLH锁原理
抢锁线程在队列尾部加入一个节点,然后仅在前驱节点上进行普通自旋,它不断轮询前一个节点状态,如果发现前一个节点释放锁,当前节点抢锁成功。
CLH抢占锁的过程
创建一个的Node,将其中的locked设置为true表示需要获取锁 线程对tail域调用getAndSet方法,使自己成为队列的尾部,同时获取一个指向其前趋结点的引用 该线程就在前趋结点的locked字段上旋转,直到前趋结点释放锁 当一个线程需要释放锁时,将当前结点的locked域设置为false,同时回收前趋结点
这篇关于Java基础知识总结(79)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!