1. 什么情況下對(duì)象會(huì)直接分配到老年代?
答: 大對(duì)象直接進(jìn)入老年代
大對(duì)象一般是指需要大量連續(xù)內(nèi)存空間的Java對(duì)象茄袖,最典型的大對(duì)象就是那種很長的字符串或數(shù)組余赢,例如new byte[10 * 1024 * 1024]是一個(gè)10M的大對(duì)象。遇到大對(duì)象對(duì)虛擬機(jī)來說是可怕的丰辣,更可怕的是遇到一群朝生夕滅的短命大對(duì)象袍镀。經(jīng)常遇到大對(duì)象评疗,會(huì)導(dǎo)致在內(nèi)存還有不少空間時(shí)就提前觸發(fā)垃圾收集,用以獲取足夠的連續(xù)內(nèi)存空間來安置大對(duì)象滞诺。
虛擬機(jī)提供了一個(gè)參數(shù)-XX:PretenureSizeThreshold形导,大于這個(gè)參數(shù)的對(duì)象會(huì)直接在老年代分配。這樣能避免在Eden區(qū)和兩個(gè)Survivor區(qū)之間大量的內(nèi)存復(fù)制(新生代采用復(fù)制算法)习霹。
2. JVM中符號(hào)引用和直接引用什么?
3. Java 中讀取文件的根目錄怎么確認(rèn)朵耕?classpath?
4. 對(duì)象的內(nèi)存布局?
5. Java對(duì)象的大小如何計(jì)算淋叶?
6. ClassNotFoundException vs. NoClassDefFoundError
7. NoSuchMethodException?
8. wait notify方法為什么必須在同步塊中阎曹?
答: 這里要說一個(gè)專業(yè)名詞:競態(tài)條件。什么是競太條件呢煞檩?
當(dāng)兩個(gè)線程競爭同一資源時(shí)处嫌,如果對(duì)資源的訪問順序敏感,就稱存在競態(tài)條件形娇。
競態(tài)條件會(huì)導(dǎo)致程序在并發(fā)情況下出現(xiàn)一些bugs锰霜。多線程對(duì)一些資源的競爭的時(shí)候就會(huì)產(chǎn)生競態(tài)條件,如果首先要執(zhí)行的程序競爭失敗排到后面執(zhí)行了桐早,那么整個(gè)程序就會(huì)出現(xiàn)一些不確定的bugs癣缅。這種bugs很難發(fā)現(xiàn)而且會(huì)重復(fù)出現(xiàn),這是因?yàn)榫€程間會(huì)隨機(jī)競爭哄酝。
假設(shè)有2個(gè)線程友存,分別是生產(chǎn)者和消費(fèi)者,他們有各自的任務(wù)陶衅。
1.1 生產(chǎn)者檢查條件(如緩存滿了)-> 1.2生產(chǎn)者必須等待
2.1 消費(fèi)者消費(fèi)了一個(gè)單位的緩存 -> 2.2重新設(shè)置了條件(如緩存沒滿) -> 2.3調(diào)用notifyAll()喚醒生產(chǎn)者
我們希望的順序是: 1.1->1.2->2.1->2.2->2.3
但是由于CPU執(zhí)行是隨機(jī)的屡立,可能會(huì)導(dǎo)致 2.3 先執(zhí)行,1.2 后執(zhí)行搀军,這樣就會(huì)導(dǎo)致生產(chǎn)者永遠(yuǎn)也醒不過來了膨俐!
所以我們必須對(duì)流程進(jìn)行管理勇皇,也就是同步,通過在同步塊中并結(jié)合 wait 和 notify 方法焚刺,我們可以手動(dòng)對(duì)線程的執(zhí)行順序進(jìn)行調(diào)整敛摘。
9. 為什么java.util.concurrent 包里沒有并發(fā)的ArrayList實(shí)現(xiàn)?
答: 我認(rèn)為在java.util.concurrent包中沒有加入并發(fā)的ArrayList實(shí)現(xiàn)的主要原因是:很難去開發(fā)一個(gè)通用并且沒有并發(fā)瓶頸的線程安全的List乳愉。
像ConcurrentHashMap這樣的類的真正價(jià)值(The real point / value of classes)并不是它們保證了線程安全兄淫。而在于它們?cè)诒WC線程安全的同時(shí)不存在并發(fā)瓶頸。舉個(gè)例子蔓姚,ConcurrentHashMap采用了鎖分段技術(shù)和弱一致性的Map迭代器去規(guī)避并發(fā)瓶頸捕虽。
所以問題在于,像“Array List”這樣的數(shù)據(jù)結(jié)構(gòu)坡脐,你不知道如何去規(guī)避并發(fā)的瓶頸泄私。拿contains() 這樣一個(gè)操作來說,當(dāng)你進(jìn)行搜索的時(shí)候如何避免鎖住整個(gè)list挨措?
另一方面挖滤,Queue 和Deque (基于Linked List)有并發(fā)的實(shí)現(xiàn)是因?yàn)樗麄兊慕涌谙啾萀ist的接口有更多的限制,這些限制使得實(shí)現(xiàn)并發(fā)成為可能浅役。
CopyOnWriteArrayList是一個(gè)有趣的例子斩松,它規(guī)避了只讀操作(如get/contains)并發(fā)的瓶頸,但是它為了做到這點(diǎn)觉既,在修改操作中做了很多工作和修改可見性規(guī)則惧盹。 此外,修改操作還會(huì)鎖住整個(gè)List瞪讼,因此這也是一個(gè)并發(fā)瓶頸钧椰。所以從理論上來說,CopyOnWriteArrayList并不算是一個(gè)通用的并發(fā)List符欠。