都在收藏的Java程序員常見(jiàn)面試題庫(kù)

都在收藏的Java程序員常見(jiàn)面試題庫(kù)

長(zhǎng)沙一度軟件培訓(xùn)      2022-03-16 14:20:01     11

都在收藏的Java程序員常見(jiàn)面試題庫(kù),  1. synchronized和reentrantlock異同  相同點(diǎn)  都實(shí)現(xiàn)了多線程同步和內(nèi)存可見(jiàn)性語(yǔ)義  都是可重入鎖  不同點(diǎn)  實(shí)現(xiàn)

課程價(jià)格 請(qǐng)咨詢

上課時(shí)段: 授課校區(qū):

詳細(xì)介紹



  1. synchronized和reentrantlock異同


  相同點(diǎn)


  都實(shí)現(xiàn)了多線程同步和內(nèi)存可見(jiàn)性語(yǔ)義


  都是可重入鎖


  不同點(diǎn)


  實(shí)現(xiàn)機(jī)制不同 synchronized通過(guò)java對(duì)象頭鎖標(biāo)記和Monitor對(duì)象實(shí)現(xiàn) reentrantlock通過(guò)CAS、ASQ(AbstractQueuedSynchronizer)和locksupport(用于阻塞和解除阻塞)實(shí)現(xiàn) synchronized依賴jvm內(nèi)存模型保證包含共享變量的多線程內(nèi)存可見(jiàn)性 reentrantlock通過(guò)ASQ的volatile state保證包含共享變量的多線程內(nèi)存可見(jiàn)性


  使用方式不同 synchronized可以修飾實(shí)例方法(鎖住實(shí)例對(duì)象)、靜態(tài)方法(鎖住類(lèi)對(duì)象)、代碼塊(顯示指定鎖對(duì)象) reentrantlock顯示調(diào)用trylock()/lock()方法,需要在finally塊中釋放鎖


  功能豐富程度不同 reentrantlock提供有限時(shí)間等候鎖(設(shè)置過(guò)期時(shí)間)、可中斷鎖(lockInterruptibly)、condition(提供await、signal等方法)等豐富語(yǔ)義 reentrantlock提供公平鎖和非公平鎖實(shí)現(xiàn) synchronized不可設(shè)置等待時(shí)間、不可被中斷(interrupted)


  2. concurrenthashmap為何讀不用加鎖


  jdk1.7


  1)HashEntry中的key、hash、next 均為final 型,只能表頭插入、刪除結(jié)點(diǎn)


  2)HashEntry類(lèi)的value域被聲明為volatile型


  3)不允許用null作為鍵和值,當(dāng)讀線程讀到某個(gè)HashEntry的 value域的值為null時(shí),便知道產(chǎn)生了沖突——發(fā)生了重排序現(xiàn)象(put設(shè)置新value對(duì)象的字節(jié)碼指令重排序),需要加鎖后重新讀入這個(gè)value值


  4)volatile變量count協(xié)調(diào)讀寫(xiě)線程之間的內(nèi)存可見(jiàn)性,寫(xiě)操作后修改count,讀操作先讀count,根據(jù)happen-before傳遞性原則寫(xiě)操作的修改讀操作能夠看到


  jdk1.8


  1)Node的val和next均為volatile型


  2)tabAt和casTabAt對(duì)應(yīng)的unsafe操作實(shí)現(xiàn)了volatile語(yǔ)義


  3. ContextClassLoader(線程上下文類(lèi)加載器)的作用


  越過(guò)類(lèi)加載器的雙親委派機(jī)制去加載類(lèi),如serviceloader實(shí)現(xiàn)


  使用線程上下文類(lèi)加載器加載類(lèi),要注意保證多個(gè)需要通信的線程間的類(lèi)加載器應(yīng)該是同一個(gè),防止因?yàn)椴煌念?lèi)加載器導(dǎo)致類(lèi)型轉(zhuǎn)換異常(ClassCastException)


  4. tomcat 類(lèi)加載機(jī)制



  不同應(yīng)用使用不同的 webapp類(lèi)加載器,實(shí)現(xiàn)應(yīng)用隔離的效果,webapp類(lèi)加載器下面是jsp類(lèi)加載器


  不同應(yīng)用共享的jar包可以放到Shared類(lèi)加載器/shared目錄下


  5. osgi類(lèi)加載機(jī)制



  osgi類(lèi)加載模型是網(wǎng)狀的,可以在模塊(Bundle)間互相委托


  osgi實(shí)現(xiàn)模塊化熱部署的關(guān)鍵是自定義類(lèi)加載器機(jī)制的實(shí)現(xiàn),每個(gè)Bundle都有一個(gè)自己的類(lèi)加載器,當(dāng)需要更換一個(gè)Bundle時(shí),就把Bundle連同類(lèi)加載器一起換掉以實(shí)現(xiàn)代碼的熱替換


  當(dāng)收到類(lèi)加載請(qǐng)求時(shí),osgi將按照下面的順序進(jìn)行類(lèi)搜索:


  1)將以java.*開(kāi)頭的類(lèi)委派給父類(lèi)加載器加載


  2)否則,將委派列表名單(配置文件org.osgi.framework.bootdelegation中定義)內(nèi)的類(lèi)委派給父類(lèi)加載器加載


  3)否則,檢查是否在import-Package中聲明,如果是,則委派給Export這個(gè)類(lèi)的Bundle的類(lèi)加載器加載


  4)否則,檢查是否在Require-Bundle中聲明,如果是,則將類(lèi)加載請(qǐng)求委托給required bundle的類(lèi)加載器


  5)否則,查找當(dāng)前Bundle的ClassPath,使用自己的類(lèi)加載器加載


  6)否則,查找類(lèi)是否在自己的Fragment Bundle中,如果在,則委派給Fragment Bundle的類(lèi)加載器加載


  7)否則,查找Dynamic import-Package(Dynamic import只有在真正用到此Package的時(shí)候才進(jìn)行加載)的Bundle,委派給對(duì)應(yīng)Bundle的類(lèi)加載器加載


  8)否則,類(lèi)查找失敗


  6. 如何結(jié)束一個(gè)一直運(yùn)行的線程


  使用退出標(biāo)志,這個(gè)flag變量要多線程可見(jiàn)


  使用interrupt,結(jié)合isInterrupted()使用


  7. threadlocal使用場(chǎng)景及問(wèn)題


  threadlocal并不能解決多線程共享變量的問(wèn)題,同一個(gè) threadlocal所包含的對(duì)象,在不同的thread中有不同的副本,互不干擾


  用于存放線程上下文變量,方便同一線程對(duì)變量的前后多次讀取,如事務(wù)、數(shù)據(jù)庫(kù)connection連接,在web編程中使用的更多


  問(wèn)題: 注意線程池場(chǎng)景使用threadlocal,因?yàn)閷?shí)際變量值存放在了thread的threadlocalmap類(lèi)型變量中,如果該值沒(méi)有remove,也沒(méi)有先set的話,可能會(huì)得到以前的舊值


  問(wèn)題: 注意線程池場(chǎng)景下的內(nèi)存泄露,雖然threadlocal的get/set會(huì)清除key(key為threadlocal的弱引用,value是強(qiáng)引用,導(dǎo)致value不釋放)為null的entry,但是最好remove


  8. 線程池從啟動(dòng)到工作的流程


  剛創(chuàng)建時(shí),里面沒(méi)有線程


  調(diào)用 execute() 添加任務(wù)時(shí):


  1)如果正在運(yùn)行的線程數(shù)量小于核心參數(shù)corePoolSize,繼續(xù)創(chuàng)建線程運(yùn)行這個(gè)任務(wù)


  2)否則,如果正在運(yùn)行的線程數(shù)量大于或等于corePoolSize,將任務(wù)加入到阻塞隊(duì)列中


  3)否則,如果隊(duì)列已滿,同時(shí)正在運(yùn)行的線程數(shù)量小于核心參數(shù)maximumPoolSize,繼續(xù)創(chuàng)建線程運(yùn)行這個(gè)任務(wù)


  4)否則,如果隊(duì)列已滿,同時(shí)正在運(yùn)行的線程數(shù)量大于或等于 maximumPoolSize,根據(jù)設(shè)置的拒絕策略處理


  5)完成一個(gè)任務(wù),繼續(xù)取下一個(gè)任務(wù)處理


  6)沒(méi)有任務(wù)繼續(xù)處理,線程被中斷或者線程池被關(guān)閉時(shí),線程退出執(zhí)行,如果線程池被關(guān)閉,線程結(jié)束


  7)否則,判斷線程池正在運(yùn)行的線程數(shù)量是否大于核心線程數(shù),如果是,線程結(jié)束,否則線程阻塞。因此線程池任務(wù)全部執(zhí)行完成后,繼續(xù)留存的線程池大小為corePoolSize


  9. 阻塞隊(duì)列BlockingQueue take和poll區(qū)別


  poll(time):取走BlockingQueue里排在首位的對(duì)象,若不能立即取出,則可以等time參數(shù)規(guī)定的時(shí)間,取不到時(shí)返回null


  take():取走BlockingQueue里排在首位的對(duì)象,若BlockingQueue為空,阻塞直到BlockingQueue有新的對(duì)象被加入


  10. 如何從FutureTask不阻塞獲取結(jié)果


  get(long timeout,TimeUnit unit),超時(shí)則返回


  輪詢,先通過(guò)isDone()判斷是否結(jié)束,然后調(diào)用get()



      以上就是長(zhǎng)沙一度軟件培訓(xùn)Java培訓(xùn)機(jī)構(gòu)小編介紹的“都在收藏的Java程序員常見(jiàn)面試題庫(kù)”的內(nèi)容,希望對(duì)大家有幫助,如有疑問(wèn),請(qǐng)?jiān)诰€咨詢,有專業(yè)老師隨時(shí)為你服務(wù)。


相關(guān)推薦


最新最全java面試題及答案(初級(jí)到高級(jí))


史上最全的中高級(jí)JAVA工程師面試題及答案匯總


Java高級(jí)開(kāi)發(fā)工程師面試題


2019史上最全java面試題題庫(kù)大全800題


哪有資深java工程師面試題


Java筆試題

培訓(xùn)啦提醒您:交易時(shí)請(qǐng)核實(shí)對(duì)方資質(zhì),對(duì)于過(guò)大宣傳或承諾需謹(jǐn)慎!任何要求預(yù)付定金、匯款等方式均存在風(fēng)險(xiǎn),謹(jǐn)防上當(dāng)。