更新時(shí)間:2019-12-16 14:42:12 來(lái)源:動(dòng)力節(jié)點(diǎn) 瀏覽2380次
很多人在找工作之前都會(huì)做不少的準(zhǔn)備,有一份全面細(xì)致的面試題會(huì)給大家減少很多麻煩,所以準(zhǔn)備找工作的同學(xué)可以參考看看:
ThreadLocal(線程變量副本)
Synchronized實(shí)現(xiàn)內(nèi)存共享,ThreadLocal為每個(gè)線程維護(hù)一個(gè)本地變量。
采用空間換時(shí)間,它用于線程間的數(shù)據(jù)隔離,為每一個(gè)使用該變量的線程提供一個(gè)副本,每個(gè)線程都可以獨(dú)立地改變自己的副本,而不會(huì)和其他線程的副本沖突。
ThreadLocal類中維護(hù)一個(gè)Map,用于存儲(chǔ)每一個(gè)線程的變量副本,Map中元素的鍵為線程對(duì)象,而值為對(duì)應(yīng)線程的變量副本。
ThreadLocal在Spring中發(fā)揮著巨大的作用,在管理Request作用域中的Bean、事務(wù)管理、任務(wù)調(diào)度、AOP等模塊都出現(xiàn)了它的身影。
Spring中絕大部分Bean都可以聲明成Singleton作用域,采用ThreadLocal進(jìn)行封裝,因此有狀態(tài)的Bean就能夠以singleton的方式在多線程中正常工作了。
“你能不能談?wù)劊琷ava GC是在什么時(shí)候,對(duì)什么東西,做了什么事情?”
在什么時(shí)候:
1、新生代有一個(gè)Eden區(qū)和兩個(gè)survivor區(qū),首先將對(duì)象放入Eden區(qū),如果空間不足就向其中的一個(gè)survivor區(qū)上放,如果仍然放不下就會(huì)引發(fā)一次發(fā)生在新生代的minor GC,將存活的對(duì)象放入另一個(gè)survivor區(qū)中,然后清空Eden和之前的那個(gè)survivor區(qū)的內(nèi)存。在某次GC過(guò)程中,如果發(fā)現(xiàn)仍然又放不下的對(duì)象,就將這些對(duì)象放入老年代內(nèi)存里去。
2、大對(duì)象以及長(zhǎng)期存活的對(duì)象直接進(jìn)入老年區(qū)。
3、當(dāng)每次執(zhí)行minor GC的時(shí)候應(yīng)該對(duì)要晉升到老年代的對(duì)象進(jìn)行分析,如果這些馬上要到老年區(qū)的老年對(duì)象的大小超過(guò)了老年區(qū)的剩余大小,那么執(zhí)行一次Full GC以盡可能地獲得老年區(qū)的空間。
對(duì)什么東西:
從GC Roots搜索不到,而且經(jīng)過(guò)一次標(biāo)記清理之后仍沒(méi)有復(fù)活的對(duì)象。
做什么:
新生代:復(fù)制清理; 老年代:標(biāo)記-清除和標(biāo)記-壓縮算法; 永久代:存放Java中的類和加載類的類加載器本身。
GC Roots都有哪些: 1. 虛擬機(jī)棧中的引用的對(duì)象 2. 方法區(qū)中靜態(tài)屬性引用的對(duì)象,常量引用的對(duì)象 3. 本地方法棧中JNI(即一般說(shuō)的Native方法)引用的對(duì)象。
Synchronized 與Loc
Synchronized 與Lock都是可重入鎖,同一個(gè)線程再次進(jìn)入同步代碼的時(shí)候。可以使用自己已經(jīng)獲取到的鎖。
Synchronized是悲觀鎖機(jī)制,獨(dú)占鎖。而Locks.ReentrantLock是,每次不加鎖而是假設(shè)沒(méi)有沖突而去完成某項(xiàng)操作,如果因?yàn)闆_突失敗就重試,直到成功為止。
ReentrantLock適用場(chǎng)景
1、某個(gè)線程在等待一個(gè)鎖的控制權(quán)的這段時(shí)間需要中斷;
2、需要分開處理一些wait-notify,ReentrantLock里面的Condition應(yīng)用,能夠控制notify哪個(gè)線程,鎖可以綁定多個(gè)條件;
3、具有公平鎖功能,每個(gè)到來(lái)的線程都將排隊(duì)等候。
StringBuffer是線程安全的,每次操作字符串,String會(huì)生成一個(gè)新的對(duì)象,而StringBuffer不會(huì);StringBuilder是非線程安全的。
fail-fast是什么?
fail-fast:機(jī)制是java集合(Collection)中的一種錯(cuò)誤機(jī)制。當(dāng)多個(gè)線程對(duì)同一個(gè)集合的內(nèi)容進(jìn)行操作時(shí),就可能會(huì)產(chǎn)生fail-fast事件。
例如:當(dāng)某一個(gè)線程A通過(guò)iterator去遍歷某集合的過(guò)程中,若該集合的內(nèi)容被其他線程所改變了;那么線程A訪問(wèn)集合時(shí),就會(huì)拋出ConcurrentModificationException異常,產(chǎn)生fail-fast事件。
happens-before
happens-before:如果兩個(gè)操作之間具有 happens-before 關(guān)系,那么前一個(gè)操作的結(jié)果就會(huì)對(duì)后面一個(gè)操作可見。
1、程序順序規(guī)則:一個(gè)線程中的每個(gè)操作,happens- before 于該線程中的任意后續(xù)操作。
2、監(jiān)視器鎖規(guī)則:對(duì)一個(gè)監(jiān)視器鎖的解鎖,happens- before 于隨后對(duì)這個(gè)監(jiān)視器鎖的加鎖。
3、volatile變量規(guī)則:對(duì)一個(gè)volatile域的寫,happens- before于任意后續(xù)對(duì)這個(gè)volatile域的讀。
4、傳遞性:如果A happens- before B,且B happens- before C,那么A happens- before C。
5、線程啟動(dòng)規(guī)則:Thread對(duì)象的start()方法happens- before于此線程的每一個(gè)動(dòng)作。
Volatile和Synchronized的不同點(diǎn)
Volatile和Synchronized四個(gè)不同點(diǎn):
1、粒度不同,前者針對(duì)變量 ,后者鎖對(duì)象和類;
2、syn阻塞,volatile線程不阻塞;
3、syn保證三大特性,volatile不保證原子性;
4、syn編譯器優(yōu)化,volatile不優(yōu)化 volatile具備兩種特性:
保證此變量對(duì)所有線程的可見性,指一條線程修改了這個(gè)變量的值,新值對(duì)于其他線程來(lái)說(shuō)是可見的,但并不是多線程安全的;
禁止指令重排序優(yōu)化。
Volatile如何保證內(nèi)存可見性:
1、當(dāng)寫一個(gè)volatile變量時(shí),JMM會(huì)把該線程對(duì)應(yīng)的本地內(nèi)存中的共享變量刷新到主內(nèi)存。
2、當(dāng)讀一個(gè)volatile變量時(shí),JMM會(huì)把該線程對(duì)應(yīng)的本地內(nèi)存置為無(wú)效。線程接下來(lái)將從主內(nèi)存中讀取共享變量。
同步:就是一個(gè)任務(wù)的完成需要依賴另外一個(gè)任務(wù),只有等待被依賴的任務(wù)完成后,依賴任務(wù)才能完成。
異步:不需要等待被依賴的任務(wù)完成,只是通知被依賴的任務(wù)要完成什么工作,只要自己任務(wù)完成了就算完成了,被依賴的任務(wù)是否完成會(huì)通知回來(lái)。(異步的特點(diǎn)就是通知)。 打電話和發(fā)短信來(lái)比喻同步和異步操作。
阻塞:CPU停下來(lái)等一個(gè)慢的操作完成以后,才會(huì)接著完成其他的工作。
非阻塞:非阻塞就是在這個(gè)慢的執(zhí)行時(shí),CPU去做其他工作,等這個(gè)慢的完成后,CPU才會(huì)接著完成后續(xù)的操作。
非阻塞會(huì)造成線程切換增加,增加CPU的使用時(shí)間能不能補(bǔ)償系統(tǒng)的切換成本需要考慮。
CAS
CAS(Compare And Swap) 無(wú)鎖算法: CAS是樂(lè)觀鎖技術(shù),當(dāng)多個(gè)線程嘗試使用CAS同時(shí)更新同一個(gè)變量時(shí),只有其中一個(gè)線程能更新變量的值,而其它線程都失敗,失敗的線程并不會(huì)被掛起,而是被告知這次競(jìng)爭(zhēng)中失敗,并可以再次嘗試。
CAS有3個(gè)操作數(shù),內(nèi)存值V,舊的預(yù)期值A(chǔ),要修改的新值B。當(dāng)且僅當(dāng)預(yù)期值A(chǔ)和內(nèi)存值V相同時(shí),將內(nèi)存值V修改為B,否則什么都不做。
類加載器工作機(jī)制
裝載:將Java二進(jìn)制代碼導(dǎo)入jvm中,生成Class文件。
連接:
a)校驗(yàn):檢查載入Class文件數(shù)據(jù)的正確性;
b)準(zhǔn)備:給類的靜態(tài)變量分配存儲(chǔ)空間;
c)解析:將符號(hào)引用轉(zhuǎn)成直接引用。
初始化:對(duì)類的靜態(tài)變量,靜態(tài)方法和靜態(tài)代碼塊執(zhí)行初始化工作。
雙親委派模型:類加載器收到類加載請(qǐng)求,首先將請(qǐng)求委派給父類加載器完成,用戶自定義加載器->應(yīng)用程序加載器->擴(kuò)展類加載器->啟動(dòng)類加載器。
Redis數(shù)據(jù)結(jié)構(gòu)
String—字符串(key-value 類型)
Hash—字典(hashmap) Redis的哈希結(jié)構(gòu)可以使你像在數(shù)據(jù)庫(kù)中更新一個(gè)屬性一樣只修改某一項(xiàng)屬性值
List—列表 實(shí)現(xiàn)消息隊(duì)列
Set—集合 利用唯一性
Sorted Set—有序集合 可以進(jìn)行排序 可以實(shí)現(xiàn)數(shù)據(jù)持久化
索引:B+,B-,全文索引
Mysql的索引是一個(gè)數(shù)據(jù)結(jié)構(gòu),旨在使數(shù)據(jù)庫(kù)高效的查找數(shù)據(jù)。
常用的數(shù)據(jù)結(jié)構(gòu)是B+Tree,每個(gè)葉子節(jié)點(diǎn)不但存放了索引鍵的相關(guān)信息還增加了指向相鄰葉子節(jié)點(diǎn)的指針,這樣就形成了帶有順序訪問(wèn)指針的B+Tree,做這個(gè)優(yōu)化的目的是提高不同區(qū)間訪問(wèn)的性能。
什么時(shí)候使用索引:
1、經(jīng)常出現(xiàn)在group by,order by和distinc關(guān)鍵字后面的字段。
2、經(jīng)常與其他表進(jìn)行連接的表,在連接字段上應(yīng)該建立索引。
3、經(jīng)常出現(xiàn)在Where子句中的字段。
4、經(jīng)常出現(xiàn)用作查詢選擇的字段。
Spring IOC
Spring支持三種依賴注入方式,分別是屬性(Setter方法)注入,構(gòu)造注入和接口注入。
在Spring中,那些組成應(yīng)用的主體及由Spring IOC容器所管理的對(duì)象被稱之為Bean。
Spring的IOC容器通過(guò)反射的機(jī)制實(shí)例化Bean并建立Bean之間的依賴關(guān)系。
簡(jiǎn)單地講,Bean就是由Spring IOC容器初始化、裝配及被管理的對(duì)象。
獲取Bean對(duì)象的過(guò)程,首先通過(guò)Resource加載配置文件并啟動(dòng)IOC容器,然后通過(guò)getBean方法獲取bean對(duì)象,就可以調(diào)用他的方法。
Spring Bean的作用域:
Singleton:Spring IOC容器中只有一個(gè)共享的Bean實(shí)例,一般都是Singleton作用域。
Prototype:每一個(gè)請(qǐng)求,會(huì)產(chǎn)生一個(gè)新的Bean實(shí)例。
Request:每一次http請(qǐng)求會(huì)產(chǎn)生一個(gè)新的Bean實(shí)例。
以上就是動(dòng)力節(jié)點(diǎn)Java培訓(xùn)機(jī)構(gòu)小編介紹的“2020年最新干貨,Java面試題總結(jié)”的內(nèi)容,希望對(duì)大家有幫助,如有疑問(wèn),請(qǐng)?jiān)诰€咨詢,有專業(yè)老師隨時(shí)為你服務(wù)。
相關(guān)推薦
最新最全java面試題及答案(初級(jí)到高級(jí))
相關(guān)閱讀
0基礎(chǔ) 0學(xué)費(fèi) 15天面授
有基礎(chǔ) 直達(dá)就業(yè)
業(yè)余時(shí)間 高薪轉(zhuǎn)行
工作1~3年,加薪神器
工作3~5年,晉升架構(gòu)
提交申請(qǐng)后,顧問(wèn)老師會(huì)電話與您溝通安排學(xué)習(xí)