天天看點

來自一位5年java程式員整理的大資料面試題與參考答案。

1.String 和StringBuffer的差別

JAVA平台提供了兩個類:String和StringBuffer,它們可以儲存和操作字元串,即包含多個字元的字元資料這個String類提供了數值不可改變字元串而這個StringBuffer類提供的字元串進行修改當你知道字元資料要改變的時候你就可以使用StringBuffer典型地,你可以使用StringBuffers來動态構造字元資料

2.說出ArrayList,Vector, LinkedList的存儲性能和特性

ArrayList和Vector都是使用數組方式存儲資料,此數組元素數大于實際存儲資料以便增加和插入元素,它們都允許直接按序号索引元素,但是插入元素要涉及數組元素移動等記憶體操作,是以索引資料快而插入資料慢,Vector由于使用了synchronized思路方法(線程安全) ,通常性能上較ArrayList差,而LinkedList使用雙向連結清單實作存儲,按序号索引資料需要進行前向或後向周遊,但是插入資料時隻需要記錄本項前後項即可,是以插入速度較快

3.HashMap和Hashtable的差別

HashMap是Hashtable的輕量級實作(非線程安全的實作),他們都完成了Map接口,主要差別在于HashMap允許空(null)鍵值(key),由于非線程安全,效率上可能高于HashtableJAVA中文站社群門戶oC)| g| ax ,HashMap允許将Null作為一個entryde key 或者value,而Hashtable不允許 HashMap把Hashtable的contains思路方法去掉了,改成containsvalue和containsKey因為contains思路方法容易讓人引起誤解Hashtable繼承自Dictionary類,而HashMap是Java1.2引進的Map interface的一個實作。

最大的差別是,Hashtable的思路方法是Synchronize的,而HashMap不是,在多個線程通路Hashtable時,不需要自己為它的思路方法實作同步,而HashMap 就必須為的提供外同步。

Hashtable和HashMap采用的hash/rehash算法都大概一樣,是以性能不會有很大的差異。

4. final, finally, finalize的差別

final 用于聲明屬性,方法和類,分别表示屬性不可變,方法不可覆寫,類不可繼承。

finally是異常處理語句結構的一部分,表示總是執行。

finalize是Object類的一個方法,在垃圾收集器執行的時候會調用被回收對象的此方法,可以覆寫此方法提供垃圾收集時的其他資源回收,例如關閉檔案等。

5. swtich是否能作用在byte上,是否能作用在long上,是否能作用在String上?

swtich中隻能用byte short int char

6. 多線程有幾種實作方法,都是什麼?哪一種方式比較優秀?同步有幾種實作方法,都是什麼?

多線程有兩種實作方法,分别是繼承Thread類與實作Runnable接口

實作Runnable接口比較好,因為實作類可以實作多個接口,而隻能繼承一個類。

同步的實作方面有兩種,分别是synchronized,wait與notify

7. java中有幾種類型的流?JDK為每種類型的流提供了一些抽象類以供繼承,請說出他們分别是哪些類

位元組流,字元流。位元組流繼承與inputStream/outputStream,字元流繼承于inputStreamReader/outputSteamWriter。

8:如果main方法被聲明為private會怎樣?

答案:能正常編譯,但運作的時候會提示”main方法不是public的”。

9:Java裡的傳引用和傳值的差別是什麼?

答案:傳引用是指傳遞的是位址而不是值本身,傳值則是傳遞值的一份拷貝。

10:如果要重寫一個對象的equals方法,還要考慮什麼?

答案:hashCode。

11:Java的”一次編寫,處處運作”是如何實作的?

答案:Java程式會被編譯成位元組碼組成的class檔案,這些位元組碼可以運作在任何平台,是以Java是平台獨立的。

12:說明一下public static void main(String args[])這段聲明裡每個關鍵字的作用

答案:public: main方法是Java程式運作時調用的第一個方法,是以它必須對Java環境可見。是以可見性設定為pulic.

static: Java平台調用這個方法時不會建立這個類的一個執行個體,是以這個方法必須聲明為static。

void: main方法沒有傳回值。

String是指令行傳進參數的類型,args是指指令行傳進的字元串數組。

13:==與equals的差別

答案:==比較兩個對象在記憶體裡是不是同一個對象,就是說在記憶體裡的存儲位置一緻。兩個String對象存儲的值是一樣的,但有可能在記憶體裡存儲在不同的地方 .

比較的是引用而equals方法比較的是内容。public boolean equals(Object obj) 這個方法是由Object對象提供的,可以由子類進行重寫。預設的實作隻有當對象和自身進行比較時才會傳回true,這個時候和是等價的。String, BitSet, Date, 和File都對equals方法進行了重寫,對兩個String對象 而言,值相等意味着它們包含同樣的字元序列。對于基本類型的包裝類來說,值相等意味着對應的基本類型的值一樣。

public class EqualsTest {
               public static void main(String[] args) {
                               String s1 = “abc”;
                               String s2 = s1;
                               String s5 = “abc”;
                               String s3 = new String(”abc”);
                               String s4 = new String(”abc”);
                               System.out.println(”== comparison : ” + (s1 == s5));
                               System.out.println(”== comparison : ” + (s1 == s2));
                               System.out.println(”Using equals method : ” + s1.equals(s2));
                               System.out.println(”== comparison : ” + s3 == s4);
                               System.out.println(”Using equals method : ” + s3.equals(s4));
               }
}
           

結果:

== comparison : true
== comparison : true
Using equals method : true
false
Using equals method :true
           

14:如果去掉了main方法的static修飾符會怎樣?

答案:程式能正常編譯。運作時會抛NoSuchMethodError異常。

15:為什麼oracle type4驅動被稱作瘦驅動?

答案:oracle提供了一個type 4 JDBC驅動,被稱為瘦驅動。這個驅動包含了一個oracle自己完全用Java實作的一個TCP/IP的Net8的實作,是以它是平台獨立的,可以在運作時由浏覽器下載下傳,不依賴任何用戶端 的oracle實作。用戶端連接配接字元串用的是TCP/IP的位址端口,而不是資料庫名的tnsname。

16:介紹一下finalize方法

答案: final: 常量聲明。 finally: 處理異常。 finalize: 幫助進行垃圾回收。

接口裡聲明的變量預設是final的。final類無法繼承,也就是沒有子類。這麼做是出于基礎類型的安全考慮,比如String和Integer。這樣也使得編譯器進行一些優化,更容易保證線程的安全性。final方法無法重寫。final變量的值不能改變。finalize()方法在一個對象被銷毀和回收前會被調用。finally,通常用于異常處理,不管有沒有異常被抛出都會執行到。比如,關閉連接配接通常放到finally塊中完成。

17:什麼是Java API?

答案:Java API是大量軟體元件的集合,它們提供了大量有用的功能,比如GUI元件。

18:GregorianCalendar類是什麼東西?

答案:GregorianCalendar提供了西方傳統月曆的支援。

19:ResourceBundle類是什麼?

答案:ResourceBundle用來存儲指定語言環境的資源,應用程式可以根據運作時的語言環境來加載這些資源,進而提供不同語言的展示。

20:為什麼Java裡沒有全局變量?

答案:全局變量是全局可見的,Java不支援全局可見的變量,因為:全局變量破壞了引用透明性原則。全局變量導緻了命名空間的沖突。

21:如何将String類型轉化成Number類型?

答案:Integer類的valueOf方法可以将String轉成Number。下面是代碼示例:

String numString = "1000";
int id=Integer.valueOf(numString).intValue();
           

22:SimpleTimeZone類是什麼?

答案:SimpleTimeZone提供公曆日期支援。

23:while循環和do循環有什麼不同?

答案:while結構在循環的開始判斷下一個疊代是否應該繼續。do/while結構在循環的結尾來判斷是否将繼續下一輪疊代。do結構至少會執行一次循環體。

24:Locale類是什麼?

答案:Locale類用來根據語言環境來動态調整程式的輸出。

25:面向對象程式設計的原則是什麼?

答案:主要有三點,多态,繼承和封裝。

26:介紹下繼承的原則

答案:繼承使得一個對象可以擷取另一個對象的屬性。使用繼承可以讓已經測試完備的功能得以複用,并且可以一次修改,所有繼承的地方都同時生效。

27:什麼是隐式的類型轉化?

答案:隐式的類型轉化就是簡單的一個類型指派給另一個類型,沒有顯式的告訴編譯器發生了轉化。并不是所有的類型都支援隐式的類型轉化。

代碼示例:

int i = 1000;

long j = i; //Implicit casting

28:sizeof是Java的關鍵字嗎?

答案:不是。

29:native方法是什麼?

答案:native方法是非Java代碼實作的方法。

30:在System.out.println()裡面,System, out, println分别是什麼?

答案:System是系統提供的預定義的final類,out是一個PrintStream對象,println是out對象裡面一個重載的方法。

31:封裝,繼承和多态是什麼?

答案:簡單來說,多态是指一個名字多種實作。多态使得一個實體通過一個通用的方式來實作不同的操作。具體的操作是由實際的實作來決定的。

多态在Java裡有三種表現方式:方法重載通過繼承實作方法重寫通過Java接口進行方法重寫。

32:顯式的類型轉化是什麼?

答案:顯式的類型轉化是明确告訴了編譯器來進行對象的轉化。

代碼示例:

long i = 700.20;
int j = (int) i; //Explicit casting
           

33:什麼是Java虛拟機?

答案:Java虛拟機是能移植到不同硬體平台上的軟體系統。

34:類型向下轉換是什麼?

答案:向下轉換是指由一個通用類型轉換成一個具體的類型,在繼承結構上向下進行。

35:Java的通路修飾符是什麼?

答案:通路權限修飾符是表明類成員的通路權限類型的關鍵字。使用這些關鍵字來限定程式的方法或者變量的通路權限。它們包含:

public: 所有類都可以通路 protected: 同一個包内以及所有子類都可以通路 private: 隻有歸屬的類才能通路預設: 歸屬類及相同包下的子類可以通路

36:所有類的父類是什麼?

答案:Object.

37:Java的基本類型有哪些?

答案:byte,char, short, int, long, float, double, boolean。

38:靜态類型有什麼特點?

答案:靜态變量是和類綁定到一起的,而不是類的執行個體對象。每一個執行個體對象都共享同樣一份靜态變量。也就是說,一個類的靜态變量隻有一份,不管它有多少個對象。類變量或者說靜态變量是通過static這個關鍵字來聲明的。類變量通常被用作常量。靜态變量通常通過類名字來進行通路。當程式運作的時候這個變量就會建立直到程式結束後才會被銷毀。類變量的作用域和執行個體變量是一樣的。它的初始值和成員變量也是一樣的,當變量沒被初始化的時候根據它的資料類型,會有一個預設值。類似的,靜态方法是屬于類的方法,而不是類對象,它的調用并不作用于類對象,也不需要建立任何的類執行個體。靜态方法本身就是final的,因為重寫隻會發生在類執行個體上,靜态方法是和類綁定在一起的,不是對象。父類的靜态方法會被子類的靜态方法屏蔽,隻要原來方法沒有聲明為final。非靜态方法不能重寫靜态方法,也就是說,你不能在子類中把一個靜态方法改成執行個體方法。

非靜态變量在每一個對象執行個體上都有單獨的一份值。

39:&操作符和&&操作符有什麼差別?

答案:當一個&表達式在求值的時候,兩個操作數都會被求值,&&更像是一個操作符的快捷方式。當一個&&表達式求值的時候,先計算第一個操作數,如果它傳回true才會計算第二個操作數。如果第一個操作數取值為fale,第二個操作數就不會被求值。

40:Java是如何處理整型的溢出和下溢的?

答案:Java根據類型的大小,将計算結果中的對應低階位元組存儲到對應的值裡面。

41:public static void寫成static public void會怎樣?

答案:程式正常編譯及運作。

42,聲明變量和定義變量有什麼不同?

答案:聲明變量我們隻提供變量的類型和名字,并沒有進行初始化。定義包括聲明和初始化兩個階段String s;隻是變量聲明,String s = new String(“bob”); 或者String s = “bob”;是變量定義。

43:Java支援哪種參數傳遞類型?

答案:Java參數都是進行傳值。對于對象而言,傳遞的值是對象的引用,也就是說原始引用和參數引用的那個拷貝,都是指向同一個對象。

44:對象封裝的原則是什麼?

答案:封裝是将資料及操作資料的代碼綁定到一個獨立的單元。這樣保障了資料的安全,防止外部代碼的錯誤使用。對象允許程式和資料進行封裝,以減少潛在的幹涉。對封裝的另一個了解是作為資料及代碼的保護層,防止保護層外代碼的随意通路。

45:你怎麼了解變量?

答案:變量是一塊命名的記憶體區域,以便程式進行通路。變量用來存儲資料,随着程式的執行,存儲的資料也可能跟着改變。

46:數值提升是什麼?

答案:數值提升是指資料從一個較小的資料類型轉換成為一個更大的資料類型,以便進行整型或者浮點型運算。在數值提升的過程中,byte,char,short值會被轉化成int類型。需要的時候int類型也可能被提升成long。long和float則有可能會被轉換成double類型。

47:Java的類型轉化是什麼?

答案:從一個資料類型轉換成另一個資料類型叫做類型轉換。Java有兩種類型轉換的方式,一個是顯式的類型轉換,一個是隐式的。

48:main方法的參數裡面,字元串數組的第一個參數是什麼?

答案:數組是空的,沒有任何元素。不像C或者C++,第一個元素預設是程式名。如果指令行沒有提供任何參數的話,main方法中的String數組為空,但不是null。

49:怎麼判斷數組是null還是為空?

答案:輸出array.length的值,如果是0,說明數組為空。如果是null的話,會抛出空指針異常。

50:程式中可以允許多個類同時擁有都有main方法嗎?

答案:可以。當程式運作的時候,我們會指定運作的類名。JVM隻會在你指定的類中查找main方法。是以多個類擁有main方法并不存在命名沖突的問題。

51:靜态變量在什麼時候加載?編譯期還是運作期?靜态代碼塊加載的時機呢?

答案:當類加載器将類加載到JVM中的時候就會建立靜态變量,這跟對象是否建立無關。靜态變量加載的時候就會配置設定記憶體空間。靜态代碼塊的代碼隻會在類第一次初始化的時候執行一次。一個類可以有多個靜态代碼塊,它并不是類的成員,也沒有傳回值,并且不能直接調用。靜态代碼塊不能包含this或者super,它們通常被用初始化靜态變量。

52:一個類能擁有多個main方法嗎?

答案:可以,但隻能有一個main方法擁有以下簽名:

否則程式将無法通過編譯。編譯器會警告你main方法已經存在。

53:簡單的介紹下JVM是如何工作的?

答案:JVM是一台抽象的計算機,就像真實的計算機那樣,它們會先将.java檔案編譯成.class檔案(.class檔案就是位元組碼檔案),然後用它的解釋器來加載位元組碼。

54:如果原地交換兩個變量的值?

答案:先把兩個值相加指派給第一個變量,然後用得到的結果減去第二個變量,指派給第二個變量。再用第一個變量減去第二個變量,同時指派給第一個變量。代碼如下:

使用異或操作也可以交換。第一個方法還可能會引起溢出。異或的方法如下: int a=5,b=10;a=a+b; b=a-b; a=a-b;

int a = 5; int b = 10;
a = a ^ b;
b = a ^ b;
a = a ^ b;
           

55:什麼是資料的封裝?

答案:資料封裝的一種方式是在類中建立set和get方法來通路對象的資料變量。一般來說變量是private的,而get和set方法是public的。封裝還可以用來在存儲資料時進行資料驗證,或者對資料進行計算,或者用作自省(比如在struts中使用javabean)。把資料和功能封裝到一個獨立的結構中稱為資料封裝。封裝其實就是把資料和關聯的操作方法封裝到一個獨立的單元中,這樣使用關聯的這些方法才能對資料進行通路操作。封裝提供的是資料安全性,它其實就是一種隐藏資料的方式。

56:什麼是反射API?它是如何實作的?

答案:反射是指在運作時能檢視一個類的狀态及特征,并能進行動态管理的功能。這些功能是通過一些内建類的反射API提供的,比如Class,Method,Field, Constructors等。使用的例子:使用Java反射API的getName方法可以擷取到類名。

57:JVM自身會維護緩存嗎,是不是在堆中進行對象配置設定,作業系統的堆還是JVM自己管理的堆?為什麼?

答案:是的,JVM自身會管理緩存,它在堆中建立對象,然後在棧中引用這些對象。

58:虛拟記憶體是什麼?

答案:虛拟記憶體又叫延伸記憶體,實際上并不存在真實的實體記憶體。

問題:方法可以同時即是static又是synchronized的嗎?

答案:可以。如果這樣做的話,JVM會擷取和這個對象關聯的java.lang.Class執行個體上的鎖。這樣做等于:

synchronized(XYZ.class) {
}
           

59:String和StringTokenizer的差別是什麼?

答案:StringTokenizer是一個用來分割字元串的工具類。

StringTokenizer st = new StringTokenizer(”Hello World”);
while (st.hasMoreTokens()) {
    System.out.println(st.nextToken());
}
           

輸出:

Hello
World
           

60:transient變量有什麼特點?

答案:transient變量不會進行序列化。例如一個實作Serializable接口的類在序列化到ObjectStream的時候,transient類型的變量不會被寫入流中,同時,反序列化回來的時候,對應變量的值為null。

61:哪些容器使用Border布局作為它們的預設布局?

答案:Window, Frame, Dialog。

62:怎麼了解什麼是同步?

答案:同步用來控制共享資源在多個線程間的通路,以保證同一時間内隻有一個線程能通路到這個資源。在非同步保護的多線程程式裡面,一個線程正在修改一個共享變量的時候,可能有另一個線程也在使用或者更新它的值。同步避免了髒資料的産生。

對方法進行同步:

public synchronized void Method1 () {
// Appropriate method-related code.
}
           

在方法内部對代碼塊進行同步:

public myFunction (){
    synchronized (this) {
            // Synchronized code here.
         }
}
           
Sun HotSpot VM,是JDK和Open JDK中自帶的虛拟機,也是目前使用範圍最廣的Java虛拟機。

JVM記憶體分布

  • 程式計數器:是一塊較小的記憶體空間,可以看作是目前線程所執行的位元組碼的行号訓示器。程式中的分支、循環、跳轉、異常處理、線程 - 恢複等基礎功能都需要依賴這個計數器完成。由于多線程是通過線程輪流切換并配置設定處理器執行時間的方式來實作的,故該區域為線程私有的記憶體。
  • 虛拟機棧:描述的是Java方法執行的記憶體模型,用于存儲局部變量表、操作數棧、動态連結、方法出口等

    堆:是Java虛拟機所管理的記憶體中最大的一塊,Java堆是被所有線程共享的一塊記憶體區域,在虛拟機啟動時建立,存放所執行個體,也是垃圾收集器管理的主要

  • 方法區:用于存放已被虛拟機加載的類資訊、常量、靜态變量、即時編譯後的代碼等資料。HotSVM針對該區域也進行GC,主要是常量回收以及類

JVM記憶體配置設定政策

對象的記憶體配置設定,在大方向上,是在Java堆上進行配置設定。

大多數情況下,對象在新生代Eden區中配置設定,當Eden區沒有足夠空間進行配置設定時,虛拟機将發起一次Minor GC。

大多數情況下,大對象直接進入老年代,虛拟機提供了參數來定義大對象的閥值,超過閥值的對象都會直接進入老年代。

經過多次Minor GC後仍然存活的對象(長期存活的對象),将進入老年代。虛拟機提供了參數,可以設定閥值。

JVM垃圾回收算法

标記-清除算法:首先标記出所有需要回收的對象,在标記完成後統一回收所有被标記的對象。

複制算法:将可用記憶體按容量劃分為大小相等的兩塊,每次隻使用其中的一塊。當一塊記憶體用完了,将還存另外一塊上面,然後在把已使用過的記憶體空間一次清理掉。

标記-整理算法:标記過程與“标記-清除”算法一樣,但後續步驟不是直接對可回收對象進行清理,而是讓所一端移動,然後直接清理掉端邊界以外的記憶體。

分代收集算法:一般是把Java堆分為新生代和老年代,根據各個年代的特點采用最适當的收集算法。新生代都發現有大批對象死去,選用複制算法。老年代中因為對象存活率高,必須使用“标記-清理”或“标記-整理”算法來進行回收。

垃圾收集器

Serial收集器:是一個單線程的收集器,隻會使用一個CPU或一條收集線程去完成垃圾收集工作,在進行垃圾收集時,必須暫停其他所有的工作線程,直到它收集結束。

ParNew收集器:是Serial收集器的多線程版本,除了使用多條線程進行垃圾收集之外,其餘行為與Serial收集器完全一樣。

CMS收集器:是一種以擷取最短回收停頓時間為目标的收集器。過程分為以下四個步驟:

  • 初始标記
               
  • 并發标記
               
  • 重新标記
               
  • 并發清除
               

JVM常見啟動參數

  • Xms / -Xmx — 堆的初始大小 / 堆的最大大小
  • Xmn — 堆中年輕代的大小
  • XX:-DisableExplicitGC — 讓System.gc()不産生任何作用
  • XX:+PrintGCDetails — 列印GC的細節
  • XX:+PrintGCDateStamps — 列印GC操作的時間戳
  • XX:NewSize / XX:MaxNewSize — 設定新生代大小/新生代最大大小
  • XX:NewRatio — 可以設定老生代和新生代的比例
  • XX:PrintTenuringDistribution — 設定每次新生代GC後輸出幸存者樂園中對象年齡的分布
  • XX:InitialTenuringThreshold / -XX:MaxTenuringThreshold:設定老年代閥值的初始值和最大值
  • XX:TargetSurvivorRatio:設定幸存區的目标使用率

JAVA類生命周期

Java類從被加載到虛拟機記憶體中開始,到解除安裝出記憶體為止,它的整個生命周期包括:加載、驗證、準備、解析、初始化、使用、解除安裝七個階段。

JVM類加載

啟動(Bootstrap)類加載器:是用本地代碼實作的類裝入器,它負責将 <Java_Runtime_Home>/lib下面的類庫加載到記憶體中(比如rt.jar)。由于引導類加載器涉及到虛拟機本地實作細節,開發者無法直接擷取到啟動類加載器的引用,是以不允許直接通過引用進行操作。

标準擴充(Extension)類加載器:是由 Sun 的 ExtClassLoader(sun.misc.Launcher E x t C l a s s L o a d e r ) 實 現 J a v a R u n t i m e H o m e > / l i b / e x t j a v a . e x t . d i r 指 定 位 置 中 的 類 庫 加 載 到 内 存 中 。 開 發 者 可 以 直 接 使 用 标 準 擴 展 類 加 載 器 。 系 統 ( S y s t e m ) 類 加 載 器 : 是 由 S u n 的 A p p C l a s s L o a d e r ( s u n . m i s c . L a u n c h e r ExtClassLoader)實作Java_Runtime_Home >/lib/extjava.ext.dir指定位置中的類庫加載到記憶體中。開發者可以直接使用标準擴充類加載器。 系統(System)類加載器:是由 Sun 的 AppClassLoader(sun.misc.Launcher ExtClassLoader)實作JavaR​untimeH​ome>/lib/extjava.ext.dir指定位置中的類庫加載到記憶體中。開發者可以直接使用标準擴充類加載器。系統(System)類加載器:是由Sun的AppClassLoader(sun.misc.LauncherAppClassLoader)實作的。徑(CLASSPATH)中指定的類庫加載到記憶體中。開發者可以直接使用系統類加

雙親委派機制描述 :某個特定的類加載器在接到加載類的請求時,首先将加載任務委托給父類加載器,依次遞歸,如果父類加載器可以完成類加載任務,就成功傳回;隻有父類加載器無法完成此加載任務時,才自己去加載。

JVM調優

檢視堆空間大小配置設定(年輕代、年老代、持久代配置設定)

垃圾回收監控(長時間監控回收情況)

  • 線程資訊監控:系統線程數量
  • 線程狀态監控:各個線程都處在什麼樣的狀态下
  • 線程詳細資訊:檢視線程内部運作情況,死鎖檢查
  • CPU熱點:檢查系統哪些方法占用了大量CPU時間
  • 記憶體熱點:檢查哪些對象在系統中數量最大

總結

針對最近很多人都在面試,我這邊也整理了相當多的面試專題資料,也有其他大廠的面經。希望可以幫助到大家。

最新整理面試題

有需要的小夥伴可以加群1149778920 暗号:qf

來自一位5年java程式員整理的大資料面試題與參考答案。
真實面試經曆
來自一位5年java程式員整理的大資料面試題與參考答案。
最新整理大廠面試文檔
來自一位5年java程式員整理的大資料面試題與參考答案。

以上就是本文的全部内容,希望對大家的學習有所幫助,也希望大家多多支援。一鍵三連哦!

來自一位5年java程式員整理的大資料面試題與參考答案。