一線網際網路常見的14個Java面試題,你顫抖了嗎程式設計師
跳槽不算頻繁,但參加過不少面試(電話面試、face to face面試),面過大/小公司、網際網路/傳統軟體公司,麵糊過(眼高手低,缺乏實戰經驗,掛掉),也面過人,所幸未因失敗而氣餒,在此過程中不斷查缺補漏,養成了踏實、追本溯源、持續改進的習慣,特此將自己經歷過、構思過的一些面試題記錄下來,如果答案有問題,歡迎拍磚討論,希望能對找工作或者感興趣的同學有所幫助,陸續整理中。
1. synchronized和reentrantlock異同
相同點
都實現了多執行緒同步和記憶體可見性語義
都是可重入鎖
不同點
實現機制不同 synchronized通過java物件頭鎖標記和Monitor物件實現 reentrantlock通過CAS、ASQ(AbstractQueuedSynchronizer)和locksupport(用於阻塞和解除阻塞)實現 synchronized依賴jvm記憶體模型保證包含共享變數的多執行緒記憶體可見性 reentrantlock通過ASQ的volatile state保證包含共享變數的多執行緒記憶體可見性
使用方式不同 synchronized可以修飾例項方法(鎖住例項物件)、靜態方法(鎖住類物件)、程式碼塊(顯示指定鎖物件) reentrantlock顯示呼叫trylock()/lock()方法,需要在finally塊中釋放鎖
功能豐富程度不同 reentrantlock提供有限時間等候鎖(設定過期時間)、可中斷鎖(lockInterruptibly)、condition(提供await、signal等方法)等豐富語義 reentrantlock提供公平鎖和非公平鎖實現 synchronized不可設定等待時間、不可被中斷(interrupted)
2. concurrenthashmap為何讀不用加鎖
jdk1.7
1)HashEntry中的key、hash、next 均為final 型,只能表頭插入、刪除結點
2)HashEntry類的value域被宣告為volatile型
3)不允許用null作為鍵和值,當讀執行緒讀到某個HashEntry的 value域的值為null時,便知道產生了衝突——發生了重排序現象(put設定新value物件的位元組碼指令重排序),需要加鎖後重新讀入這個value值
4)volatile變數count協調讀寫執行緒之間的記憶體可見性,寫操作後修改count,讀操作先讀count,根據happen-before傳遞性原則寫操作的修改讀操作能夠看到
jdk1.8
1)Node的val和next均為volatile型
2)tabAt和casTabAt對應的unsafe操作實現了volatile語義
3. ContextClassLoader(執行緒上下文類載入器)的作用
越過類載入器的雙親委派機制去載入類,如serviceloader實現
使用執行緒上下文類載入器載入類,要注意保證多個需要通訊的執行緒間的類載入器應該是同一個,防止因為不同的類載入器導致型別轉換異常(ClassCastException)
4. tomcat 類載入機制
不同應用使用不同的 webapp類載入器,實現應用隔離的效果,webapp類載入器下面是jsp類載入器
不同應用共享的jar包可以放到Shared類載入器/shared目錄下
5. osgi類載入機制
osgi類載入模型是網狀的,可以在模組(Bundle)間互相委託
osgi實現模組化熱部署的關鍵是自定義類載入器機制的實現,每個Bundle都有一個自己的類載入器,當需要更換一個Bundle時,就把Bundle連同類載入器一起換掉以實現程式碼的熱替換
當收到類載入請求時,osgi將按照下面的順序進行類搜尋:
1)將以java.*開頭的類委派給父類載入器載入
2)否則,將委派列表名單(配置檔案org.osgi.framework.bootdelegation中定義)內的類委派給父類載入器載入
3)否則,檢查是否在Import-Package中宣告,如果是,則委派給Export這個類的Bundle的類載入器載入
4)否則,檢查是否在Require-Bundle中宣告,如果是,則將類載入請求委託給required bundle的類載入器
5)否則,查詢當前Bundle的ClassPath,使用自己的類載入器載入
6)否則,查詢類是否在自己的Fragment Bundle中,如果在,則委派給Fragment Bundle的類載入器載入
7)否則,查詢Dynamic Import-Package(Dynamic Import只有在真正用到此Package的時候才進行載入)的Bundle,委派給對應Bundle的類載入器載入
8)否則,類查詢失敗
6. 如何結束一個一直執行的執行緒
使用退出標誌,這個flag變數要多執行緒可見
使用interrupt,結合isInterrupted()使用
7. threadlocal使用場景及問題
threadlocal並不能解決多執行緒共享變數的問題,同一個 threadlocal所包含的物件,在不同的thread中有不同的副本,互不干擾
用於存放執行緒上下文變數,方便同一執行緒對變數的前後多次讀取,如事務、資料庫connection連線,在web程式設計中使用的更多
問題: 注意執行緒池場景使用threadlocal,因為實際變數值存放在了thread的threadlocalmap型別變數中,如果該值沒有remove,也沒有先set的話,可能會得到以前的舊值
問題: 注意執行緒池場景下的記憶體洩露,雖然threadlocal的get/set會清除key(key為threadlocal的弱引用,value是強引用,導致value不釋放)為null的entry,但是最好remove
8. 執行緒池從啟動到工作的流程
剛建立時,裡面沒有執行緒
呼叫 execute() 新增任務時:
1)如果正在執行的執行緒數量小於核心引數corePoolSize,繼續建立執行緒執行這個任務
2)否則,如果正在執行的執行緒數量大於或等於corePoolSize,將任務加入到阻塞佇列中
3)否則,如果佇列已滿,同時正在執行的執行緒數量小於核心引數maximumPoolSize,繼續建立執行緒執行這個任務
4)否則,如果佇列已滿,同時正在執行的執行緒數量大於或等於 maximumPoolSize,根據設定的拒絕策略處理
5)完成一個任務,繼續取下一個任務處理
6)沒有任務繼續處理,執行緒被中斷或者執行緒池被關閉時,執行緒退出執行,如果執行緒池被關閉,執行緒結束
7)否則,判斷執行緒池正在執行的執行緒數量是否大於核心執行緒數,如果是,執行緒結束,否則執行緒阻塞。因此執行緒池任務全部執行完成後,繼續留存的執行緒池大小為corePoolSize
9. 阻塞佇列BlockingQueue take和poll區別
poll(time):取走BlockingQueue裡排在首位的物件,若不能立即取出,則可以等time引數規定的時間,取不到時返回null
take():取走BlockingQueue裡排在首位的物件,若BlockingQueue為空,阻塞直到BlockingQueue有新的物件被加入
10. 如何從FutureTask不阻塞獲取結果
get(long timeout,TimeUnit unit),超時則返回
輪詢,先通過isDone()判斷是否結束,然後呼叫get()
11. blockingqueue如果存放了比較關鍵的資料,系統當機該如何處理
開放性問題,歡迎討論
將佇列持久化,比較麻煩,需要將生產資料持久化到磁碟,持久化成功才返回,消費者執行緒從磁碟載入資料到記憶體阻塞佇列中,維護消費offset,啟動時,根據消費offset從磁碟載入資料
加入訊息佇列,保證訊息不丟失,生成序列號,消費冪等,根據消費程式決定系統重啟後的生產狀態
12. NIO與傳統I/O的區別
節約執行緒,NIO由原來的每個執行緒都需要阻塞讀寫變成了由單執行緒(即Selector)負責處理多個channel註冊(register)的興趣事件(SelectionKey)集合(底層藉助作業系統提供的epoll()),netty bossgroup處理accept連線(沒看明白為什麼bossgroup設定多個thread的必要性),workergroup處理具體業務流程和資料讀寫
NIO提供非阻塞操作
傳統I/O 以流的方式處理資料,而 NIO 以塊的方式處理資料,NIO提供bytebuffer,分為堆內和堆外緩衝區,讀寫時均先放到該緩衝區中,然後由核心通過channel傳輸到對端,堆外緩衝區不走核心,提升了效能
13. list中存放可重複字串,如何刪除某個字串
呼叫iterator相關方法刪除
倒刪,防止正序刪除導致的陣列重排,index跳過陣列元素問題
14. 有哪些GC ROOTS(跟日常開發比較相關的是和此相關的記憶體洩露)
所有Java執行緒當前活躍的棧幀裡指向GC堆裡的物件的引用,因此用不到的物件及時置null,提升記憶體回收效率
靜態變數引用的物件,因此減少靜態變數特別是靜態集合變數的大小,集合存放的物件覆寫euqls()和hashcode(),防止持續增長
本地方法JNI引用的物件
方法區中的常量引用的物件,因此減少在長字串上呼叫String.intern()
classloader載入的class物件,因此自定義classloader無效時及時置null並且注意類載入器載入物件之間的隔離
jvm裡的一些靜態資料結構裡指向GC堆裡的物件的引用
…
微信公眾號“演算法數學之美”,由演算法與數學之美團隊打造的另一個公眾號,歡迎大家掃碼關注!
更多精彩:
演算法數學之美微信公眾號歡迎賜稿
稿件涉及數學、物理、演算法、計算機、程式設計等相關領域,經採用我們將奉上稿酬。
投稿郵箱:math_alg@163.com
相關文章
- 11月13日雲棲精選夜讀|一線網際網路常見的14個Java面試題Java面試題
- 網際網路 Java 工程師面試題(Spring 面試題一)Java工程師面試題Spring
- 好程式設計師Java學習路線分享Spring常見面試題程式設計師JavaSpring面試題
- Java程式設計師面試常見問題Java程式設計師面試
- Linux網路程式設計常見面試題Linux程式設計面試題
- 程式設計師面試常問計算機網路問題程式設計師面試計算機網路
- 好程式設計師Java教程分享JavaScript常見面試題一程式設計師JavaScript面試題
- 好程式設計師分享:Java面試題常見問題程式設計師Java面試題
- 好程式設計師Java教程分享:Java工程師常見面試題程式設計師Java工程師面試題
- 好程式設計師Java教程分享XML常見面試題程式設計師JavaXML面試題
- 好程式設計師Java教程分享JavaScript常見面試題五程式設計師JavaScript面試題
- 好程式設計師Java教程分享JavaScript常見面試題四程式設計師JavaScript面試題
- 好程式設計師Java教程分享JavaScript常見面試題三程式設計師JavaScript面試題
- 好程式設計師Java教程分享JavaScript常見面試題二程式設計師JavaScript面試題
- 5年程式設計師面試,常見面試問題解析程式設計師面試
- 好程式設計師Java教程分享Java面試常見技術難題程式設計師Java面試
- [面試專題]一線網際網路大廠面試總結面試
- 程式設計師如何實現“網際網路+”01-程式設計師走向自由職業的 5 種常見方式程式設計師
- 一個老程式設計師在網際網路寒冬下的感悟程式設計師
- 好程式設計師web前端學習路線分享HTML5常見面試題程式設計師Web前端HTML面試題
- 好程式設計師web前端分享常見面試題程式設計師Web前端面試題
- 2018年一線網際網路公司Java高階面試題總結Java面試題
- 好程式設計師Java教程分享Java多執行緒常見面試題程式設計師Java執行緒面試題
- 網路安全常見面試題-Web方向面試題Web
- 一線網際網路企業面試題總結(幫你成功拿到offer)面試題
- 一線網際網路大廠MySQL索引面試真題MySql索引面試
- 好程式設計師Python教程分享常見的Python面試題程式設計師Python面試題
- 【200人面試經驗】,程式設計師面試,常見面試題解析程式設計師面試題
- Java工程師常見面試題集錦(一)網際網路人必看!(附答案及視訊教程,持續更新)Java工程師面試題
- 大型網際網路公司必考java面試題與面試技巧Java面試題
- 新鮮出爐的一線網際網路公司Java高階面試題總結!Java面試題
- 好程式設計師分享Java常見面試題Tomcat最佳化經驗程式設計師Java面試題Tomcat
- JAVA學習線路:day14-網路程式設計Java程式設計
- 好程式設計師Java培訓分享Java多執行緒常見面試問題程式設計師Java執行緒面試
- 分享:一線網際網路公司的面試經驗面試
- linux c網路網路程式設計面試題收集Linux程式設計面試題
- 好程式設計師Python教程分享Python常見面試問題程式設計師Python面試
- 好程式設計師web前端教程分享Jquery常見面試題程式設計師Web前端jQuery面試題