【威哥說】昨天的文章發(fā)出后祝迂,好多同學(xué)告訴我說自己現(xiàn)在就是處在一個迷茫的時期睦尽,找不到方向。不用著急型雳,你肯定可以找到屬于自己的種子当凡,也許和我一樣是Java山害,也許是其他。你需要立刻做的就是去捫心自問沿量,是自己興趣粗恢,還是高薪的工作,抑或是你僅僅需要磨礪營作為平臺欧瘪。今天在這里和大家聊一聊在Java求職中經(jīng)常被問到的關(guān)于線程的問題眷射。
一、現(xiàn)在有T1佛掖、T2妖碉、T3三個線程,你怎樣保證T2在T1執(zhí)行完后執(zhí)行芥被,T3在T2執(zhí)行完后執(zhí)行欧宜?
答:可以使用join方法實(shí)現(xiàn),join:表示等其他線程拴魄,他有一個時間參數(shù)冗茸,也可以不設(shè)置時間參數(shù)。當(dāng)沒有的時候匹中,表示等這個線程執(zhí)行完再往下執(zhí)行夏漱,當(dāng)有參數(shù)時,假設(shè)現(xiàn)在主線程為mainThread顶捷,支線程為aThread,bThread挂绰;mainThread里執(zhí)行aThread.join(10000) ;表示mainThread等aThread執(zhí)行10秒再說。但如果10秒里服赎,aThread最多只要執(zhí)行5秒就完事葵蒂,那么,mainThread只要等5秒就會不再等了重虑。因?yàn)閍Thread已經(jīng)完事了践付。如果10秒里aThread要10秒以上的時間才能執(zhí)行完。那會怎樣呢缺厉?mainThread會不管你執(zhí)行完沒完永高,等你10秒就不等了。哥先走一步芽死。
二乏梁、在Java中Lock比synchronized塊的優(yōu)勢是什么?你需要實(shí)現(xiàn)一個高效的緩存关贵,它允許多個用戶讀遇骑,但只允許一個用戶寫,以此來保持它的完整性揖曾,你會怎樣去實(shí)現(xiàn)它落萎?
答:lock接口在多線程和并發(fā)編程中最大的優(yōu)勢是它們?yōu)樽x和寫分別提供了鎖亥啦,它能滿足你寫像ConcurrentHashMap這樣的高性能數(shù)據(jù)結(jié)構(gòu)和有條件的阻塞。Java線程面試的問題越來越會根據(jù)面試者的回答來提問练链。我強(qiáng)烈建議在你去參加多線程的面試之前認(rèn)真讀一下Locks翔脱,因?yàn)楫?dāng)前其大量用于構(gòu)建電子交易終統(tǒng)的客戶端緩存和交易連接空間。
三媒鼓、在java中wait和sleep方法的不同届吁?
答:通常會在電話面試中經(jīng)常被問到的Java線程面試問題。最大的不同是在等待時wait會釋放鎖绿鸣,而sleep一直持有鎖疚沐。Wait通常被用于線程間交互,sleep通常被用于暫停執(zhí)行潮模。
對于sleep()方法亮蛔,我們首先要知道該方法是屬于Thread類中的。而wait()方法擎厢,則是屬于Object類中的究流。
sleep()方法導(dǎo)致了程序暫停執(zhí)行指定的時間,讓出cpu該其他線程动遭,但是他的監(jiān)控狀態(tài)依然保持者芬探,當(dāng)指定的時間到了又會自動恢復(fù)運(yùn)行狀態(tài)。
在調(diào)用sleep()方法的過程中沽损,線程不會釋放對象鎖灯节。
而當(dāng)調(diào)用wait()方法的時候,線程會放棄對象鎖绵估,進(jìn)入等待此對象的等待鎖定池,只有針對此對象調(diào)用notify()方法后本線程才進(jìn)入對象鎖定池準(zhǔn)備卡骂。
四国裳、用Java寫代碼來解決生產(chǎn)者——消費(fèi)者問題。
與上面的問題很類似全跨,但這個問題更經(jīng)典缝左,有些時候面試都會問下面的問題。在Java中怎么解決生產(chǎn)者——消費(fèi)者問題浓若,當(dāng)然有很多解決方法渺杉,我已經(jīng)分享了一種用阻塞隊列實(shí)現(xiàn)的方法。有些時候他們甚至?xí)栐趺磳?shí)現(xiàn)哲學(xué)家進(jìn)餐問題挪钓。
五是越、如何預(yù)防死鎖
死鎖是這樣一種情形:多個線程同時被阻塞,它們中的一個或者全部都在等待某個資源被釋放碌上。由于線程被無限期地阻塞倚评,因此程序不可能正常終止浦徊。
導(dǎo)致死鎖的根源在于不適當(dāng)?shù)剡\(yùn)用“synchronized”關(guān)鍵詞來管理線程對特定對象的訪問√煳啵“synchronized”關(guān)鍵詞的作用是盔性,確保在某個時刻只有一個線程被允許執(zhí)行特定的代碼塊,因此呢岗,被允許執(zhí)行的線程首先必須擁有對變量或?qū)ο蟮呐潘缘脑L問權(quán)冕香。當(dāng)線程訪問對象時,線程會給對象加鎖后豫,而這個鎖導(dǎo)致其它也想訪問同一對象的線程被阻塞暂筝,直至第一個線程釋放它加在對象上的鎖。 所以我們要在使用同步的時候去關(guān)心它硬贯,避免發(fā)生焕襟。
六、為什么我們調(diào)用start()方法時會執(zhí)行run()方法饭豹,為什么我們不能直接調(diào)用run()方法鸵赖?
線程的運(yùn)行不是像其他方法那么簡單的額調(diào)用一下就運(yùn)行的。想要運(yùn)行一個線程拄衰,肯定是要和操作系統(tǒng)打交道它褪,告訴操作系統(tǒng)我要添加一個線程,run方法只是個簡單的方法翘悉,里面就是你要運(yùn)行的代碼茫打。start方法里面會調(diào)用一個native方法,這個方法才是真正和操作系統(tǒng)交流的
七妖混、Java中你怎樣喚醒一個阻塞的線程老赤?
這是個關(guān)于線程和阻塞的棘手的問題,它有很多解決方法制市。如果線程遇到了IO阻塞抬旺,我并且不認(rèn)為有一種方法可以中止線程。如果線程因?yàn)檎{(diào)用wait()祥楣、sleep()开财、或者join()方法而導(dǎo)致的阻塞,你可以中斷線程误褪,并且通過拋出InterruptedException來喚醒它责鳍。