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