ThreadLocal(線程變量副本)
Synchronized實現(xiàn)內(nèi)存共享,ThreadLocal為每個線程維護一個本地變量。
采用空間換時間,它用于線程間的數(shù)據(jù)隔離,為每一個使用該變量的線程提供一個副本,每個線程都可以獨立地改變自己的副本,而不會和其他線程的副本沖突。
ThreadLocal類中維護一個Map,用于存儲每一個線程的變量副本,Map中元素的鍵為線程對象,而值為對應(yīng)線程的變量副本。
ThreadLocal在Spring中發(fā)揮著巨大的作用,在管理Request作用域中的Bean、事務(wù)管理、任務(wù)調(diào)度、AOP等模塊都出現(xiàn)了它的身影。
Spring中絕大部分Bean都可以聲明成Singleton作用域,采用ThreadLocal進行封裝,因此有狀態(tài)的Bean就能夠以singleton的方式在多線程中正常工作了。
Java內(nèi)存模型:
Java虛擬機規(guī)范中將Java運行時數(shù)據(jù)分為六種。
1.程序計數(shù)器:是一個數(shù)據(jù)結(jié)構(gòu),用于保存當前正常執(zhí)行的程序的內(nèi)存地址。Java虛擬機的多線程就是通過線程輪流切換并分配處理器時間來實現(xiàn)的,為了線程切換后能恢復(fù)到正確的位置,每條線程都需要一個獨立的程序計數(shù)器,互不影響,該區(qū)域為“線程私有”。
2.Java虛擬機棧:線程私有的,與線程生命周期相同,用于存儲局部變量表,操作棧,方法返回值。局部變量表放著基本數(shù)據(jù)類型,還有對象的引用。
3.本地方法棧:跟虛擬機棧很像,不過它是為虛擬機使用到的Native方法服務(wù)。
4.Java堆:所有線程共享的一塊內(nèi)存區(qū)域,對象實例幾乎都在這分配內(nèi)存。
5.方法區(qū):各個線程共享的區(qū)域,儲存虛擬機加載的類信息,常量,靜態(tài)變量,編譯后的代碼。
6.運行時常量池:代表運行時每個class文件中的常量表。包括幾種常量:編譯時的數(shù)字常量、方法或者域的引用。
“你能不能談?wù)劊琷ava GC是在什么時候,對什么東西,做了什么事情?”
在什么時候:
1.新生代有一個Eden區(qū)和兩個survivor區(qū),首先將對象放入Eden區(qū),如果空間不足就向其中的一個survivor區(qū)上放,如果仍然放不下就會引發(fā)一次發(fā)生在新生代的minor GC,將存活的對象放入另一個survivor區(qū)中,然后清空Eden和之前的那個survivor區(qū)的內(nèi)存。在某次GC過程中,如果發(fā)現(xiàn)仍然又放不下的對象,就將這些對象放入老年代內(nèi)存里去。
2.大對象以及長期存活的對象直接進入老年區(qū)。
3.當每次執(zhí)行minor GC的時候應(yīng)該對要晉升到老年代的對象進行分析,如果這些馬上要到老年區(qū)的老年對象的大小超過了老年區(qū)的剩余大小,那么執(zhí)行一次Full GC以盡可能地獲得老年區(qū)的空間。
對什么東西:從GC Roots搜索不到,而且經(jīng)過一次標記清理之后仍沒有復(fù)活的對象。
做什么: 新生代:復(fù)制清理; 老年代:標記-清除和標記-壓縮算法; 永久代:存放Java中的類和加載類的類加載器本身。
GC Roots都有哪些: 1. 虛擬機棧中的引用的對象 2. 方法區(qū)中靜態(tài)屬性引用的對象,常量引用的對象 3. 本地方法棧中JNI(即一般說的Native方法)引用的對象。
Synchronized關(guān)鍵字、Lock,并解釋它們之間的區(qū)別
Synchronized 與Lock都是可重入鎖,同一個線程再次進入同步代碼的時候.可以使用自己已經(jīng)獲取到的鎖。
Synchronized是悲觀鎖機制,獨占鎖。而Locks.ReentrantLock是,每次不加鎖而是假設(shè)沒有沖突而去完成某項操作,如果因為沖突失敗就重試,直到成功為止。 ReentrantLock適用場景
某個線程在等待一個鎖的控制權(quán)的這段時間需要中斷
需要分開處理一些wait-notify,ReentrantLock里面的Condition應(yīng)用,能夠控制notify哪個線程,鎖可以綁定多個條件。
具有公平鎖功能,每個到來的線程都將排隊等候。
Volatile和Synchronized四個不同點:
1 粒度不同,前者針對變量 ,后者鎖對象和類
2 syn阻塞,volatile線程不阻塞
3 syn保證三大特性,volatile不保證原子性
4 syn編譯器優(yōu)化,volatile不優(yōu)化 volatile具備兩種特性:
保證此變量對所有線程的可見性,指一條線程修改了這個變量的值,新值對于其他線程來說是可見的,但并不是多線程安全的。
禁止指令重排序優(yōu)化。
Volatile如何保證內(nèi)存可見性:
1.當寫一個volatile變量時,JMM會把該線程對應(yīng)的本地內(nèi)存中的共享變量刷新到主內(nèi)存。
2.當讀一個volatile變量時,JMM會把該線程對應(yīng)的本地內(nèi)存置為無效。線程接下來將從主內(nèi)存中讀取共享變量。同步:就是一個任務(wù)的完成需要依賴另外一個任務(wù),只有等待被依賴的任務(wù)完成后,依賴任務(wù)才能完成。
異步:不需要等待被依賴的任務(wù)完成,只是通知被依賴的任務(wù)要完成什么工作,只要自己任務(wù)完成了就算完成了,被依賴的任務(wù)是否完成會通知回來。(異步的特點就是通知)。 打電話和發(fā)短信來比喻同步和異步操作。
阻塞:CPU停下來等一個慢的操作完成以后,才會接著完成其他的工作。
非阻塞:非阻塞就是在這個慢的執(zhí)行時,CPU去做其他工作,等這個慢的完成后,CPU才會接著完成后續(xù)的操作。
非阻塞會造成線程切換增加,增加CPU的使用時間能不能補償系統(tǒng)的切換成本需要考慮。
類加載器工作機制:
1.裝載:將Java二進制代碼導(dǎo)入jvm中,生成Class文件。
2.連接:a)校驗:檢查載入Class文件數(shù)據(jù)的正確性 b)準備:給類的靜態(tài)變量分配存儲空間 c)解析:將符號引用轉(zhuǎn)成直接引用
3:初始化:對類的靜態(tài)變量,靜態(tài)方法和靜態(tài)代碼塊執(zhí)行初始化工作。
雙親委派模型:類加載器收到類加載請求,首先將請求委派給父類加載器完成 用戶自定義加載器->應(yīng)用程序加載器->擴展類加載器->啟動類加載器。
以上就是長沙牛耳教育Java培訓(xùn)機構(gòu)小編介紹的“Java面試題解惑系列:互聯(lián)網(wǎng)企業(yè)面試篇”的內(nèi)容,希望對大家有幫助,如有疑問,請在線咨詢,有專業(yè)老師隨時為你服務(wù)。
相關(guān)推薦
最新最全java面試題及答案(初級到高級)
史上最全的中高級JAVA工程師面試題及答案匯總
Java高級開發(fā)工程師面試題
2019史上最全java面試題題庫大全800題
哪有資深java工程師面試題
Java面試題