Java并發(fā)編程之LockSupport類(lèi)詳解
一、LockSupport類(lèi)的屬性
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(); // 線(xiàn)程類(lèi)類(lèi)型 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類(lèi)的構(gòu)造函數(shù)
// 私有構(gòu)造函數(shù),無(wú)法被實(shí)例化 private LockSupport() { }
三、park(Object blocker)方法 和 park()方法分析
//調(diào)用park函數(shù)時(shí),當(dāng)前線(xiàn)程首先設(shè)置好parkBlocker字段,然后再調(diào)用 Unsafe的park函數(shù) // 此后,當(dāng)前線(xiàn)程就已經(jīng)阻塞了,等待該線(xiàn)程的unpark函數(shù)被調(diào)用,所以后面的一個(gè) setBlocker函數(shù)無(wú)法運(yùn)行 // unpark函數(shù)被調(diào)用,該線(xiàn)程獲得許可后,就可以繼續(xù)運(yùn)行了,也就運(yùn)行第二個(gè) setBlocker // 把該線(xiàn)程的parkBlocker字段設(shè)置為null,這樣就完成了整個(gè)park函數(shù)的邏輯。 // 總之,必須要保證在park(Object blocker)整個(gè)函數(shù) 執(zhí)行完后,該線(xiàn)程的parkBlocker字段又恢復(fù)為null。 //阻塞當(dāng)前線(xiàn)程,并且將當(dāng)前線(xiàn)程的parkBlocker字段設(shè)置為blocker public static void park(Object blocker) { // 獲取當(dāng)前線(xiàn)程 Thread t = Thread.currentThread(); //將當(dāng)前線(xiàn)程的parkBlocker字段設(shè)置為blocker setBlocker(t, blocker); //阻塞當(dāng)前線(xiàn)程,第一個(gè)參數(shù)表示isAbsolute,是否為絕對(duì)時(shí)間,第二個(gè)參數(shù)就是代表時(shí)間 UNSAFE.park(false, 0L); //重新可運(yùn)行后再此設(shè)置Blocker setBlocker(t, null); } //無(wú)限阻塞線(xiàn)程,直到有其他線(xiàn)程調(diào)用unpark方法 public static void park() { UNSAFE.park(false, 0L); }
四、parkNanos(Object blocker,long nanos)方法 和 parkNanos(long nanos)方法分析
//阻塞當(dāng)前線(xiàn)程nanos秒 毫秒 public static void parkNanos(Object blocker, long nanos) { //先判斷nanos是否大于0,小于等于0都代表無(wú)限等待 if (nanos > 0) { // 獲取當(dāng)前線(xiàn)程 Thread t = Thread.currentThread(); //將當(dāng)前線(xiàn)程的parkBlocker字段設(shè)置為blocker setBlocker(t, blocker); //阻塞當(dāng)前線(xiàn)程現(xiàn)對(duì)時(shí)間的nanos秒 UNSAFE.park(false, nanos); //將當(dāng)前線(xiàn)程的parkBlocker字段設(shè)置為null setBlocker(t, null); } } //阻塞當(dāng)前線(xiàn)程nanos秒 毫秒 public static void parkNanos(long nanos) { if (nanos > 0) UNSAFE.park(false, nanos); }
五、parkUntil(Object blocker,long deadline)方法 和 parkUntil(long deadline)方法分析
//將當(dāng)前線(xiàn)程阻塞絕對(duì)時(shí)間的deadline秒,并且將當(dāng)前線(xiàn)程的parkBlockerOffset設(shè)置為blocker public static void parkUntil(Object blocker, long deadline) { //獲取當(dāng)前線(xiàn)程 Thread t = Thread.currentThread(); //設(shè)置當(dāng)前線(xiàn)程parkBlocker字段設(shè)置為blocker setBlocker(t, blocker); //阻塞當(dāng)前線(xiàn)程絕對(duì)時(shí)間的deadline秒 UNSAFE.park(true, deadline); //當(dāng)前線(xiàn)程parkBlocker字段設(shè)置為null setBlocker(t, null); } //將當(dāng)前線(xiàn)程阻塞絕對(duì)時(shí)間的deadline秒 public static void parkUntil(long deadline) { UNSAFE.park(true, deadline); }
六、setBlocker(Thread t, Object arg)和 getBlocker(Thread t) 方法分析
// 設(shè)置線(xiàn)程t的parkBlocker字段的值為arg private static void setBlocker(Thread t, Object arg) { UNSAFE.putObject(t, parkBlockerOffset, arg); } //獲取當(dāng)前線(xiàn)程的Blocker值 public static Object getBlocker(Thread t) { //若當(dāng)前線(xiàn)程為空就拋出異常 if (t == null) throw new NullPointerException(); //利用unsafe對(duì)象獲取當(dāng)前線(xiàn)程的Blocker值 return UNSAFE.getObjectVolatile(t, parkBlockerOffset); }
七、unpark(Thread thread) 方法分析
//釋放該線(xiàn)程的阻塞狀態(tài),即類(lèi)似釋放鎖,只不過(guò)這里是將許可設(shè)置為1 public static void unpark(Thread thread) { // 線(xiàn)程為不空 if (thread != null) // 釋放該線(xiàn)程許可 UNSAFE.unpark(thread); }
八、LockSupport優(yōu)點(diǎn)
LockSupport比Object的wait/notify有兩大優(yōu)勢(shì)
1.LockSupport不需要在同步代碼塊里 。所以線(xiàn)程間也不需要維護(hù)一個(gè)共享的同步對(duì)象了,實(shí)現(xiàn)了線(xiàn)程間的解耦。
2.unpark函數(shù)可以先于park調(diào)用,所以不需要擔(dān)心線(xiàn)程間的執(zhí)行的先后順序。
到此這篇關(guān)于Java并發(fā)編程之LockSupport類(lèi)詳解的文章就介紹到這了,更多相關(guān)Java LockSupport類(lèi)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java實(shí)現(xiàn)簡(jiǎn)單計(jì)算器小程序
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)簡(jiǎn)單計(jì)算器小程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-07-07Spring?Boot之Validation自定義實(shí)現(xiàn)方式的總結(jié)
這篇文章主要介紹了Spring?Boot之Validation自定義實(shí)現(xiàn)方式的總結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07Spring中@Configuration注解的Full模式和Lite模式詳解
這篇文章主要介紹了Spring中@Configuration注解的Full模式和Lite模式詳解,準(zhǔn)確來(lái)說(shuō),Full?模式和?Lite?模式其實(shí)?Spring?容器在處理?Bean?時(shí)的兩種不同行為,這兩種不同的模式在使用時(shí)候的表現(xiàn)完全不同,今天就來(lái)和各位小伙伴捋一捋這兩種模式,需要的朋友可以參考下2023-09-09一文詳解如何配置MyBatis實(shí)現(xiàn)打印可執(zhí)行的SQL語(yǔ)句
在MyBatis中,動(dòng)態(tài)SQL是一個(gè)強(qiáng)大的特性,允許我們?cè)赬ML映射文件或注解中編寫(xiě)條件語(yǔ)句,根據(jù)運(yùn)行時(shí)的參數(shù)來(lái)決定SQL的具體執(zhí)行內(nèi)容,這篇文章主要給大家介紹了關(guān)于如何配置MyBatis實(shí)現(xiàn)打印可執(zhí)行的SQL語(yǔ)句的相關(guān)資料,需要的朋友可以參考下2024-08-08詳談springboot過(guò)濾器和攔截器的實(shí)現(xiàn)及區(qū)別
今天小編就為大家分享一篇詳談springboot過(guò)濾器和攔截器的實(shí)現(xiàn)及區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-08-08SpringCloud Gateway的路由,過(guò)濾器和限流解讀
這篇文章主要介紹了SpringCloud Gateway的路由,過(guò)濾器和限流解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02