通過(guò)Java多線程方式實(shí)現(xiàn)循環(huán)順序打印A、B,而且保證無(wú)論多少次循環(huán),都不亂序?
1. 問(wèn)題描述
通過(guò)Java多線程方式實(shí)現(xiàn)循環(huán)順序打印A、B,而且保證無(wú)論多少次循環(huán),都不亂序?
2. 問(wèn)題的解決方案
其實(shí)這個(gè)問(wèn)題,背后考察的是一個(gè)生產(chǎn)者和消費(fèi)者的問(wèn)題。即:要保證當(dāng)前一個(gè)線程的任務(wù)完成之后,再去執(zhí)行另一個(gè)線程的任務(wù)。有多種解決方案:
1)第一種:利用wait和notify方法和synchronized關(guān)鍵字聯(lián)合完成。
2)第二種:利用重入鎖ReentrantLock和Condition
3. 方案分析:
第一種更接近于原理層面。但是,第二種更加簡(jiǎn)單。因?yàn)橹厝腈i把這些都封裝起來(lái)了,在使用起來(lái)也更加方便。
4. 核心類介紹:
(1) 重入鎖ReentrantLock:
(2)Condition類的核心方法:
1)await() 方法會(huì)是當(dāng)前線程等待,同時(shí)釋放當(dāng)前鎖,當(dāng)其他線程中使用signal() 或signalAll() 方法時(shí),線程會(huì)重新獲得鎖并繼續(xù)執(zhí)行。或者當(dāng)線程被中斷時(shí),也能跳出等待。這和obj.wait方法很相似。
2)awaitUninterruptibly() 方法與await() 方法基本相同,只不過(guò)它不會(huì)在等待過(guò)程中響應(yīng)中斷。
3)signal() 方法用于喚醒一個(gè)在等待中的線程,這和obj.notify方法很類似。
5. 基于以上方案的核心代碼如下:
輸出結(jié)果是:
A,B
A,B
A,B
A,B
A,B
一直循環(huán)下去打印A,B