高級(jí)java工程師面試題
在高級(jí)的Java工程師看了,面試做好準(zhǔn)備很有必要,那么你了解面試題目了嗎?下面小編已經(jīng)為你們整理了高級(jí)java工程師面試題,希望可以幫到你。
高級(jí)java工程師面試題1
1)Java 中能創(chuàng)建 Volatile 數(shù)組嗎?
能,Java 中可以創(chuàng)建 volatile 類(lèi)型數(shù)組,不過(guò)只是一個(gè)指向數(shù)組的引用,而不是整個(gè)數(shù)組。我的意思是,如果改變引用指向的數(shù)組,將會(huì)受到 volatile 的保護(hù),但是如果多個(gè)線程同時(shí)改變數(shù)組的元素,volatile 標(biāo)示符就不能起到之前的保護(hù)作用了。
2)volatile 能使得一個(gè)非原子操作變成原子操作嗎?
一個(gè)典型的例子是在類(lèi)中有一個(gè)long 類(lèi)型的成員變量。如果你知道該成員變量會(huì)被多個(gè)線程訪問(wèn),如計(jì)數(shù)器、價(jià)格等,你最好是將其設(shè)置為 volatile。為什么?因?yàn)?Java 中讀取 long 類(lèi)型變量不是原子的,需要分成兩步,如果一個(gè)線程正在修改該 long 變量的值,另一個(gè)線程可能只能看到該值的一半(前 32 位)。但是對(duì)一個(gè) volatile 型的 long 或 double 變量的讀寫(xiě)是原子。
3)volatile 修飾符的有過(guò)什么實(shí)踐?
一種實(shí)踐是用 volatile 修飾 long 和 double 變量,使其能按原子類(lèi)型來(lái)讀寫(xiě)。double 和 long 都是64位寬,因此對(duì)這兩種類(lèi)型的讀是分為兩部分的,第一次讀取第一個(gè) 32 位,然后再讀剩下的 32 位,這個(gè)過(guò)程不是原子的,但 Java 中 volatile 型的 long 或 double 變量的讀寫(xiě)是原子的。volatile 修復(fù)符的另一個(gè)作用是提供內(nèi)存屏障(memory barrier),例如在分布式框架中的應(yīng)用。簡(jiǎn)單的說(shuō),就是當(dāng)你寫(xiě)一個(gè) volatile 變量之前,Java 內(nèi)存模型會(huì)插入一個(gè)寫(xiě)屏障(writebarrier),讀一個(gè) volatile 變量之前,會(huì)插入一個(gè)讀屏障(read barrier)。意思就是說(shuō),在你寫(xiě)一個(gè) volatile 域時(shí),能保證任何線程都能看到你寫(xiě)的值,同時(shí),在寫(xiě)之前,也能保證任何數(shù)值的更新對(duì)所有線程是可見(jiàn)的,因?yàn)閮?nèi)存屏障會(huì)將其他所有寫(xiě)的值更新到緩存。
4)volatile 類(lèi)型變量提供什么保證?
volatile 變量提供順序和可見(jiàn)性保證,例如,JVM或者 JIT為了獲得更好的性能會(huì)對(duì)語(yǔ)句重排序,但是 volatile類(lèi)型變量即使在沒(méi)有同步塊的情況下賦值也不會(huì)與其他語(yǔ)句重排序。 volatile 提供 happens-before 的保證,確保一個(gè)線程的修改能對(duì)其他線程是可見(jiàn)的。某些情況下,volatile 還能提供原子性,如讀 64 位數(shù)據(jù)類(lèi)型,像 long 和 double 都不是原子的,但 volatile 類(lèi)型的 double 和 long 就是原子的。
5) 10 個(gè)線程和 2 個(gè)線程的同步代碼,哪個(gè)更容易寫(xiě)?
從寫(xiě)代碼的角度來(lái)說(shuō),兩者的復(fù)雜度是相同的,因?yàn)橥酱a與線程數(shù)量是相互獨(dú)立的。但是同步策略的選擇依賴(lài)于線程的數(shù)量,因?yàn)樵蕉嗟木€程意味著更大的競(jìng)爭(zhēng),所以你需要利用同步技術(shù),如鎖分離,這要求更復(fù)雜的代碼和專(zhuān)業(yè)知識(shí)。
高級(jí)java工程師面試題2
1)哪個(gè)類(lèi)包含 clone 方法?是 Cloneable 還是 Object?
java.lang.Cloneable 是一個(gè)標(biāo)示性接口,不包含任何方法,clone 方法在 object 類(lèi)中定義。并且需要知道clone() 方法是一個(gè)本地方法,這意味著它是由 c 或 c++ 或 其他本地語(yǔ)言實(shí)現(xiàn)的。
2)Java 中 ++ 操作符是線程安全的嗎?
不是線程安全的操作。它涉及到多個(gè)指令,如讀取變量值,增加,然后存儲(chǔ)回內(nèi)存,這個(gè)過(guò)程可能會(huì)出現(xiàn)多個(gè)線程交差。
3)a = a + b 與 a += b 的區(qū)別
+= 隱式的將加操作的結(jié)果類(lèi)型強(qiáng)制轉(zhuǎn)換為持有結(jié)果的類(lèi)型。如果兩這個(gè)整型相加,如 byte、short 或者 int,首先會(huì)將它們提升到 int 類(lèi)型,然后在執(zhí)行加法操作。如果加法操作的結(jié)果比 a 的最大值要大,則 a+b 會(huì)出現(xiàn)編譯錯(cuò)誤,但是 a += b 沒(méi)問(wèn)題,如下:
byte a = 127;
byte b = 127;
b = a + b; // error : cannot convert from int to byte
b += a; // ok
(譯者注:這個(gè)地方應(yīng)該表述的有誤,其實(shí)無(wú)論 a+b的值為多少,編譯器都會(huì)報(bào)錯(cuò),因?yàn)?a+b 操作會(huì)將 a、b提升為 int 類(lèi)型,所以將 int 類(lèi)型賦值給 byte 就會(huì)編譯出錯(cuò))
4)我能在不進(jìn)行強(qiáng)制轉(zhuǎn)換的情況下將一個(gè) double 值賦值給 long 類(lèi)型的變量嗎?
不行,你不能在沒(méi)有強(qiáng)制類(lèi)型轉(zhuǎn)換的前提下將一個(gè) double 值賦值給 long 類(lèi)型的變量,因?yàn)?double 類(lèi)型的范圍比 long 類(lèi)型更廣,所以必須要進(jìn)行強(qiáng)制轉(zhuǎn)換。
5)為什么 Java 中的 String 是不可變的(Immutable)?
Java 中的 String 不可變是因?yàn)?Java 的設(shè)計(jì)者認(rèn)為字符串使用非常頻繁,將字符串設(shè)置為不可變可以允許多個(gè)客戶(hù)端之間共享相同的字符串。
高級(jí)java工程師面試題3
1.Java的HashMap是如何工作的?
HashMap是一個(gè)針對(duì)數(shù)據(jù)結(jié)構(gòu)的鍵值,每個(gè)鍵都會(huì)有相應(yīng)的值,關(guān)鍵是識(shí)別這樣的值。
HashMap 基于 hashing 原理,我們通過(guò) put ()和 get ()方法儲(chǔ)存和獲取對(duì)象。當(dāng)我們將鍵值對(duì)傳遞給 put ()方法時(shí),它調(diào)用鍵對(duì)象的 hashCode ()方法來(lái)計(jì)算 hashcode,讓后找到 bucket 位置來(lái)儲(chǔ)存值對(duì)象。當(dāng)獲取對(duì)象時(shí),通過(guò)鍵對(duì)象的 equals ()方法找到正確的鍵值對(duì),然后返回值對(duì)象。HashMap 使用 LinkedList 來(lái)解決碰撞問(wèn)題,當(dāng)發(fā)生碰撞了,對(duì)象將會(huì)儲(chǔ)存在 LinkedList 的下一個(gè)節(jié)點(diǎn)中。 HashMap 在每個(gè) LinkedList 節(jié)點(diǎn)中儲(chǔ)存鍵值對(duì)對(duì)象。
2.什么是快速失敗的故障安全迭代器?
快速失敗的Java迭代器可能會(huì)引發(fā)ConcurrentModifcationException在底層集合迭代過(guò)程中被修改。故障安全作為發(fā)生在實(shí)例中的一個(gè)副本迭代是不會(huì)拋出任何異常的??焖偈〉墓收习踩独x了當(dāng)遭遇故障時(shí)系統(tǒng)是如何反應(yīng)的。例如,用于失敗的快速迭代器ArrayList和用于故障安全的迭代器ConcurrentHashMap。
3.Java BlockingQueue是什么?
Java BlockingQueue是一個(gè)并發(fā)集合util包的一部分。BlockingQueue隊(duì)列是一種支持操作,它等待元素變得可用時(shí)來(lái)檢索,同樣等待空間可用時(shí)來(lái)存儲(chǔ)元素。
4.什么時(shí)候使用ConcurrentHashMap?
在問(wèn)題2中我們看到ConcurrentHashMap被作為故障安全迭代器的一個(gè)實(shí)例,它允許完整的并發(fā)檢索和更新。當(dāng)有大量的并發(fā)更新時(shí),ConcurrentHashMap此時(shí)可以被使用。這非常類(lèi)似于Hashtable,但ConcurrentHashMap不鎖定整個(gè)表來(lái)提供并發(fā),所以從這點(diǎn)上ConcurrentHashMap的性能似乎更好一些。所以當(dāng)有大量更新時(shí)ConcurrentHashMap應(yīng)該被使用。
5.哪一個(gè)List實(shí)現(xiàn)了最快插入?
LinkedList和ArrayList是另個(gè)不同變量列表的實(shí)現(xiàn)。ArrayList的優(yōu)勢(shì)在于動(dòng)態(tài)的增長(zhǎng)數(shù)組,非常適合初始時(shí)總長(zhǎng)度未知的情況下使用。LinkedList的優(yōu)勢(shì)在于在中間位置插入和刪除操作,速度是最快的。
LinkedList實(shí)現(xiàn)了List接口,允許null元素。此外LinkedList提供額外的get,remove,方法在LinkedList的首部或尾部。這些操作使LinkedList可被用作堆棧(stack),隊(duì)列(queue)或雙向隊(duì)列(deque)。
ArrayList實(shí)現(xiàn)了可變大小的數(shù)組。它允許所有元素,包括null。 每個(gè)ArrayList實(shí)例都有一個(gè)容量(Capacity),即用于存儲(chǔ)元素的數(shù)組的大小。這個(gè)容量可隨著不斷添加新元素而自動(dòng)增加,但是增長(zhǎng)算法并沒(méi)有定義。當(dāng)需要插入大量元素時(shí),在插入前可以調(diào)用ensureCapacity方法來(lái)增加ArrayList的容量以提高插入效率。
看了“高級(jí)java工程師面試題”