Java并發編程之LockSupport類詳解
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實例 UNSAFE = sun.misc.Unsafe.getUnsafe(); // 線程類類型 Class<?> tk = Thread.class; // 獲取Thread的parkBlocker字段的內存偏移地址 parkBlockerOffset = UNSAFE.objectFieldOffset(tk.getDeclaredField('parkBlocker')); // 獲取Thread的threadLocalRandomSeed字段的內存偏移地址 SEED = UNSAFE.objectFieldOffset(tk.getDeclaredField('threadLocalRandomSeed')); // 獲取Thread的threadLocalRandomProbe字段的內存偏移地址 PROBE = UNSAFE.objectFieldOffset(tk.getDeclaredField('threadLocalRandomProbe')); // 獲取Thread的threadLocalRandomSecondarySeed字段的內存偏移地址 SECONDARY = UNSAFE.objectFieldOffset(tk.getDeclaredField('threadLocalRandomSecondarySeed'));} catch (Exception ex) { throw new Error(ex); } }二、LockSupport類的構造函數
// 私有構造函數,無法被實例化 private LockSupport() { }三、park(Object blocker)方法 和 park()方法分析
//調用park函數時,當前線程首先設置好parkBlocker字段,然后再調用 Unsafe的park函數 // 此后,當前線程就已經阻塞了,等待該線程的unpark函數被調用,所以后面的一個 setBlocker函數無法運行 // unpark函數被調用,該線程獲得許可后,就可以繼續運行了,也就運行第二個 setBlocker // 把該線程的parkBlocker字段設置為null,這樣就完成了整個park函數的邏輯。 // 總之,必須要保證在park(Object blocker)整個函數 執行完后,該線程的parkBlocker字段又恢復為null。 //阻塞當前線程,并且將當前線程的parkBlocker字段設置為blocker public static void park(Object blocker) {// 獲取當前線程Thread t = Thread.currentThread();//將當前線程的parkBlocker字段設置為blockersetBlocker(t, blocker);//阻塞當前線程,第一個參數表示isAbsolute,是否為絕對時間,第二個參數就是代表時間UNSAFE.park(false, 0L);//重新可運行后再此設置BlockersetBlocker(t, null); } //無限阻塞線程,直到有其他線程調用unpark方法 public static void park() {UNSAFE.park(false, 0L); }四、parkNanos(Object blocker,long nanos)方法 和 parkNanos(long nanos)方法分析
//阻塞當前線程nanos秒 毫秒 public static void parkNanos(Object blocker, long nanos) {//先判斷nanos是否大于0,小于等于0都代表無限等待if (nanos > 0) { // 獲取當前線程 Thread t = Thread.currentThread(); //將當前線程的parkBlocker字段設置為blocker setBlocker(t, blocker); //阻塞當前線程現對時間的nanos秒 UNSAFE.park(false, nanos); //將當前線程的parkBlocker字段設置為null setBlocker(t, null);} } //阻塞當前線程nanos秒 毫秒 public static void parkNanos(long nanos) {if (nanos > 0) UNSAFE.park(false, nanos); }五、parkUntil(Object blocker,long deadline)方法 和 parkUntil(long deadline)方法分析
//將當前線程阻塞絕對時間的deadline秒,并且將當前線程的parkBlockerOffset設置為blocker public static void parkUntil(Object blocker, long deadline) {//獲取當前線程Thread t = Thread.currentThread();//設置當前線程parkBlocker字段設置為blockersetBlocker(t, blocker);//阻塞當前線程絕對時間的deadline秒UNSAFE.park(true, deadline);//當前線程parkBlocker字段設置為nullsetBlocker(t, null); } //將當前線程阻塞絕對時間的deadline秒 public static void parkUntil(long deadline) {UNSAFE.park(true, deadline); }六、setBlocker(Thread t, Object arg)和 getBlocker(Thread t) 方法分析
// 設置線程t的parkBlocker字段的值為arg private static void setBlocker(Thread t, Object arg) {UNSAFE.putObject(t, parkBlockerOffset, arg); } //獲取當前線程的Blocker值 public static Object getBlocker(Thread t) {//若當前線程為空就拋出異常if (t == null) throw new NullPointerException();//利用unsafe對象獲取當前線程的Blocker值return UNSAFE.getObjectVolatile(t, parkBlockerOffset); }七、unpark(Thread thread) 方法分析
//釋放該線程的阻塞狀態,即類似釋放鎖,只不過這里是將許可設置為1 public static void unpark(Thread thread) {// 線程為不空if (thread != null) // 釋放該線程許可 UNSAFE.unpark(thread); }八、LockSupport優點
LockSupport比Object的wait/notify有兩大優勢
1.LockSupport不需要在同步代碼塊里 。所以線程間也不需要維護一個共享的同步對象了,實現了線程間的解耦。
2.unpark函數可以先于park調用,所以不需要擔心線程間的執行的先后順序。
到此這篇關于Java并發編程之LockSupport類詳解的文章就介紹到這了,更多相關Java LockSupport類內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!
相關文章:
