15個Java線程并發(fā)面試題和答案

1. 現(xiàn)在有線程 T1、T2 和 T3。你如何確保 T2 線程在 T1 之后執(zhí)行,并且 T3 線程在 T2 之后執(zhí)行?
這個線程面試題通常在第一輪面試或電話面試時被問到,這道多線程問題為了測試面試者是否熟悉 join 方法的概念。答案也非常簡單——可以用 Thread 類的 join 方法實現(xiàn)這一效果。
2. Java 中新的 Lock 接口相對于同步代碼塊(synchronized block)有什么優(yōu)勢?如果讓你實現(xiàn)一個高性能緩存,支持并發(fā)讀取和單一寫入,你如何保證數(shù)據(jù)完整性。
多線程和并發(fā)編程中使用 lock 接口的最大優(yōu)勢是它為讀和寫提供兩個單獨的鎖,可以讓你構(gòu)建高性能數(shù)據(jù)結(jié)構(gòu),比如 ConcurrentHashMap 和條件阻塞。
這道 Java 線程面試題越來越多見,而且隨后的面試題都基于面試者對這道題的回答。
我強烈建議在任何 Java 多線程面試前都要多看看有關(guān)鎖的知識,因為如今電子交易系統(tǒng)的客戶端和數(shù)據(jù)交互中,鎖被頻繁使用來構(gòu)建緩存。
3. Java 中 wait 和 sleep 方法有什么區(qū)別?
我們來看看另一個經(jīng)常被問到的線程面試題。這道題常出現(xiàn)在電話面試中。兩者主要的區(qū)別就是等待釋放鎖和監(jiān)視器。sleep方法在等待時不會釋放任何鎖或監(jiān)視器。wait 方法多用于線程間通信,而 sleep 只是在執(zhí)行時暫停??梢钥次伊硪黄嘘P(guān)Java 中 wait 和 sleep的文章。
4. 如何在 Java 中實現(xiàn)一個阻塞隊列?
這是一道相對困難的 Java 多線程面試題,考察點很多。它考察了面試者是否真正寫過 Java 多線程代碼,考察了面試者對并發(fā)場景的理解。并且可以根據(jù)面試者的代碼問很多后續(xù)問題,如果他用 wait() 和 notify() 方法成功實現(xiàn)了阻塞隊列,可以讓他用 Java 5 的并發(fā)類重新實現(xiàn)一次。
5. 如何在 Java 中編寫代碼解決生產(chǎn)者消費者問題?答案
和上面有關(guān)線程的問題相似,這個問題在工作中很典型,但有時面試官會問這類問題,比如“在 Java 中如何解決生產(chǎn)者消費者問題?”其實,有很多解決方式。我分享過用 Java 中 BlockingQueue 的解決方案。有時他們甚至會讓你給出哲學家進餐問題的解決方案。
6. 寫一段死鎖代碼。你在 Java 中如何解決死鎖?
這是我最喜歡的 Java 多線程面試題,因為即使死鎖在多線程并發(fā)編程中十分常見,許多面試者仍然抓耳撓腮,不能寫出無死鎖的代碼。
只需要問他們?nèi)绻?N 個資源和 N 個線程去執(zhí)行某個操作,然后請求所有資源。
這里的 N 可以是 2 作為最簡單的情況,也可以是個很大的數(shù)字讓問題變復雜。有關(guān)死鎖的更多信息可以看這篇文章Java 中如何避免死鎖。
7. 什么是原子操作?Java 中有哪些原子操作?
這是個簡單的 Java 線程面試題。另一個緊隨其后的問題將是:你需要同步原子操作嗎?你可以看這篇文章了解更多Java 同步。
8. Java 中 volatile 關(guān)鍵字是什么?你如何使用它?它和 Java 中的同步方法有什么區(qū)別?
自從 Java 5 中調(diào)整 volatile 關(guān)鍵字和 Java 內(nèi)存模型后,有關(guān) volatile 關(guān)鍵字的線程問題越來越常見。掌握 volatile變量在并發(fā)環(huán)境中如何確保可見性、有序性和一致性非常重要。
9. 什么是競態(tài)條件?你如何發(fā)現(xiàn)并解決競態(tài)條件?
這個 Java 多線程問題一般出現(xiàn)在高級面試。多數(shù)面試官會問你最近一次遇到的競態(tài)條件,如何解決的,有時他們也會寫點簡單代碼讓你發(fā)現(xiàn)競態(tài)條件??梢钥纯次业倪@篇文章Java 中的競態(tài)條件。我認為,這是最棒的 Java 線程面試問題之一,而且可以測試出面試者解決競態(tài)條件的經(jīng)驗,或是編寫無數(shù)據(jù)競爭、無其競態(tài)條件的代碼經(jīng)驗。
10. 在 Java 中你如何轉(zhuǎn)儲線程(thread dump)?如何分析它?
在 UNIX 中,你可以使用 kill -3 然后線程轉(zhuǎn)儲日志會打印在屏幕上,可以使用 CTRL+Break 查看。這只是一個較簡單的線程面試題,狡猾一點的話他們會問你如何分析轉(zhuǎn)儲日志。線程轉(zhuǎn)儲日志對于分析死鎖情況非常有用。
11. 既然 start() 方法會調(diào)用 run() 方法,為什么我們調(diào)用 start() 方法,而不直接調(diào)用 run() 方法?
這是一個基本的 Java 多線程面試題。最初,我剛開始多線程編程時對此還有些困惑。如今我一般在 Java 中級面試的電話面試或一輪面試中遇到。
這道問題的答案是這樣的。當你調(diào)用 start() 方法時,它會新建一個線程然后執(zhí)行 run() 方法中的代碼。如果直接調(diào)用 run() 方法,并不會創(chuàng)建新線程,方法中的代碼會在當前調(diào)用者的線程中執(zhí)行。可以看這篇文章了解更多線程中 Start 和 Run 方法的區(qū)別。
Java面試:投行的15個多線程和并發(fā)面試題
12. Java 中你如何喚醒阻塞線程?
這是有關(guān)線程的一個很狡猾的問題。有很多原因會導致阻塞,如果是 IO 阻塞,我認為沒有方式可以中斷線程(如果有的話請告訴我)。另一方面,如果線程阻塞是由于調(diào)用了 wait(),sleep() 或 join() 方法,你可以中斷線程,通過拋出 InterruptedException 異常來喚醒該線程。可以看這篇文章了解有關(guān)處理阻塞線程的知識Java 中如何處理阻塞方法。
13. Java 中 CyclicBarriar 和 CountdownLatch 有什么區(qū)別?
最近的 Java 線程面試題多數(shù)在測試你對 JDK 5 并發(fā)包的掌握程度。兩者區(qū)別之一就是 CyclicBarrier 在屏障打開之后(所有線程到達屏障點),可以重復使用。而 CountDownLatch 不行。想了解更多可以參與課程Java 中的多線程和并行計算。
14. 什么是不可變類?它對于編寫并發(fā)應用有何幫助?
盡管這道面試題和線程沒有直接關(guān)系,但間接影響也很大。如果面試官隨后讓你寫一個不可變類,或問你為什么 Java 中的 String 是不可變的,會讓面試題變得更加復雜。
15. 你在多線程環(huán)境中遇到的最多的問題是什么?你如何解決的?
內(nèi)存干擾、競態(tài)條件、死鎖、活鎖、線程饑餓是多線程和并發(fā)編程中比較有代表性的問題。這類問題無休無止,而且難于定位和調(diào)試。
相關(guān)文章
Java研發(fā)面試99題(含答案):JVM+Spring+MySQL+線程池+鎖
這篇文章主要介紹了Java研發(fā)面試99題,主要包括了JVM,Spring,MySQL,線程池,鎖等,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-07-16