AQS理解之一,基础知识——LockSupport
AQS理解之一,基礎知識——LockSupport
LockSupport類位于java.util.concurrent包下。
顧名思義,就是一個實現鎖的輔助類。
來看下他的類結構:
其中的變量都是通過UNSAFE類來賦值,代碼如下:
private static final sun.misc.Unsafe UNSAFE;private static final long parkBlockerOffset;private static final long SEED;private static final long PROBE;private static final long SECONDARY;static {try {UNSAFE = sun.misc.Unsafe.getUnsafe();Class<?> tk = Thread.class;parkBlockerOffset = UNSAFE.objectFieldOffset(tk.getDeclaredField("parkBlocker"));SEED = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomSeed"));PROBE = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomProbe"));SECONDARY = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomSecondarySeed"));} catch (Exception ex) { throw new Error(ex); }}parkBlockerOffset 是Thread類中的parkBlocker參數,其他參數類似。
parkBlockerOffset參數設置之后可以在jstack的時候對給這個變量設置了值的線程進行觀察。
方法是park()和unpark(Thread t)和這兩個方法的一些帶時間參數方法。
通過看這個類的注釋,可以知道park 和unpark實際上是對線程的阻塞與解除阻塞。
park: 阻塞線程,線程在一下三種情況下會被打開:
1.調用unpark方法,釋放該線程的許可。
2.該線程被中斷。
3.時間到期。
LockSupport的park和unpark操作的實際動作。
1.調用unpark
如果沒有可用線程,則給定許可(permit就變成1(不會累計))
如果有線程被阻塞,解除鎖,同時park返回。
如果給定線程沒有啟動,則該操作不能保證有任何效果.
2.調用park,則會檢測permit是否為1。
如果為1則將permit變成0;
如果不為1,則堵塞線程,直到permit變為1.
我們知道wait和notify也可以實現對線程的阻塞和解除,他們的區別主要在于:
1,unpark會直接指定要解除阻塞的線程,而notify需要知道有一個確定的線程在wait,如果有多個線程在阻塞,則不能確定知道哪個會被解除阻塞。
2,wait和notify有先后順序,即必須先wait,再notify才能解除,而park和unpark則沒有,可以先給權限,再阻塞,阻塞會直接返回。
3,wait時線程如果被interrupt,會報錯InterruptedException,而park時則會正常結束。
4,wait/notify面向對象,而LockSupport面向線程
總結
以上是生活随笔為你收集整理的AQS理解之一,基础知识——LockSupport的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 深入浅出理解锁之—— AbstractQ
- 下一篇: AQS理解之二,自己设计一个锁