自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

竟然還有人沒搞懂 JVM 類加載器?一文徹底明白

云計(jì)算 虛擬化
在Java面試中,在考察完項(xiàng)目經(jīng)驗(yàn)、基礎(chǔ)技術(shù)后,我會根據(jù)候選人的特點(diǎn)進(jìn)行知識深度的考察,如果候選人簡歷上有寫JVM(Java虛擬機(jī))相關(guān)的東西,那么我常常會問一些JVM的問題。

[[280350]]

寫在前面

在Java面試中,在考察完項(xiàng)目經(jīng)驗(yàn)、基礎(chǔ)技術(shù)后,我會根據(jù)候選人的特點(diǎn)進(jìn)行知識深度的考察,如果候選人簡歷上有寫JVM(Java虛擬機(jī))相關(guān)的東西,那么我常常會問一些JVM的問題。JVM的類加載機(jī)制是一個(gè)很經(jīng)典的知識點(diǎn),圍繞這個(gè)知識點(diǎn)可以有下面這些難度不同的問題。

  1. 簡單講下JVM中的類加載過程
  2. JVM中的類加載和卸載的時(shí)機(jī)?
  3. 如何理解JVM中不同類加載器的概念和作用?
  4. 簡單講下JVM中的雙親委派模型?
  5. 什么情況下會破壞雙親委派模型?為什么?可否舉個(gè)例子?
  6. Tomcat中的類加載機(jī)制有了解嗎?為什么這么設(shè)計(jì)?
  7. 實(shí)際開發(fā)中有遇到哪些類加載器相關(guān)的問題?你又是如何解決的?
  8. JVM之上的弱類型語言例如Groovy是如何實(shí)現(xiàn)?簡單講下動態(tài)類加載機(jī)制?

在接下來的幾篇文章,我將跟讀者一起重新梳理一遍類加載器的相關(guān)知識,爭取能夠妥善解答上面列出的這些問題。

基本概念篇

類的加載和卸載

JVM是虛擬機(jī)的一種,它的指令集語言是字節(jié)碼,字節(jié)碼構(gòu)成的文件是class文件。平常我們寫的Java文件,需要編譯為class文件才能交給JVM運(yùn)行。可以這么說:C語言代碼——>二進(jìn)制文件——>計(jì)算機(jī)硬件,就相當(dāng)于Java代碼——>字節(jié)碼文件——>JVM。JVM將指定的class文件讀取到內(nèi)存里,并運(yùn)行該class文件里的Java程序的過程,就稱之為類的加載;反之,將某個(gè)class文件的運(yùn)行時(shí)數(shù)據(jù)從JVM中移除的過程,就稱之為類的卸載。

class文件的運(yùn)行時(shí)數(shù)據(jù)就是C++對象,也稱為kclass對象,這些運(yùn)行時(shí)數(shù)據(jù)在JDK7之前是放在永久代(PermGen),JDK8之后則放在元空間(Metaspace)。

類的生命周期

Java類從被虛擬機(jī)加載開始,到卸載出內(nèi)存為止,它的整個(gè)生命周期包括:加載(Loading)、驗(yàn)證(Verification)、準(zhǔn)備(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using)和卸載(Unloading)7個(gè)階段;其中驗(yàn)證、準(zhǔn)備和解析又統(tǒng)稱為連接(Linking)階段。

 

竟然還有人沒搞懂 JVM 類加載器?一文徹底明白

 

類的加載的時(shí)機(jī)

虛擬機(jī)規(guī)范并未嚴(yán)格規(guī)定類加載的時(shí)機(jī),跟具體的JVM虛擬機(jī)有關(guān)。類加載的最佳時(shí)機(jī)是解析Java字節(jié)碼類文件中常量池符號的時(shí)候,Class.forName()、ClassLoader.loadClass()、反射API和JNI_FindClass都可以觸發(fā)類加載,Hot JVM自身啟動的時(shí)候也會觸發(fā)類加載。

通過JVM參數(shù)中加-verbose:class,可以在應(yīng)用啟動的時(shí)候打印類加載的過程,如下圖所示:

 

竟然還有人沒搞懂 JVM 類加載器?一文徹底明白

 

  1. 初始化這個(gè)階段,JVM虛擬機(jī)給出了5種必須對類進(jìn)行“初始化”的情況
  2. 使用new關(guān)鍵字實(shí)例化對象的時(shí)候、讀取或設(shè)置一個(gè)類的靜態(tài)字段的時(shí)候、調(diào)用一個(gè)類的靜態(tài)方法的時(shí)候;
  3. 使用java.lang.reflect包的方法對類進(jìn)行反射調(diào)用的時(shí)候,如果類沒有進(jìn)行過初始化,則要先觸發(fā)其初始化;
  4. 當(dāng)初始化一個(gè)類的時(shí)候,如果發(fā)現(xiàn)其父類還沒有被初始化,則要先初始化其父類;
  5. 當(dāng)虛擬機(jī)啟動時(shí),用戶需要指定一個(gè)執(zhí)行的主類(包含main方法的那個(gè)類),則虛擬機(jī)會優(yōu)先初始化這個(gè)主類;
  6. 在JDK1.7以后,動態(tài)語言支持的時(shí)候,如果一個(gè)java.lang.invoke.MethodHandle實(shí)例最后的結(jié)果是要執(zhí)行第1種情況的操作,則也要進(jìn)行初始化。

類的卸載時(shí)機(jī)

類的卸載跟采用的垃圾收集算法有關(guān),在CMS中有兩種方法卸載不必要的類,一種是等到元空間(Metaspace)滿了的時(shí)候觸發(fā)FGC,另一種是使用跟CMS并發(fā)收集算法類似的方式,不過對于元空間的閾值和觸發(fā)CMS并發(fā)收集的閾值是獨(dú)立的。更具體的可以參考之前的文章:CMS學(xué)習(xí)筆記。在這里,我們只需要記住,JVM中一個(gè)類的卸載要滿足下面這3個(gè)條件:

  1. 該類所有的實(shí)例對象都已被回收;
  2. 該類的類加載器對象已經(jīng)被回收;
  3. 該類對應(yīng)的java.lang.Class對象沒有在任何地方被引用,無法在任何地方通過反射訪問該類的方法。

類加載器的作用

類的加載是需要類加載器完成的,但是類加載器在JVM中的作用可不止這些。在JVM中,一個(gè)類的唯一性是需要這個(gè)類本身和類加載一起才能確定的,每個(gè)類加載器都有一個(gè)獨(dú)立的命名空間。

不同的類加載器,即使是同一個(gè)類字節(jié)碼文件,最后再JVM里的類對象也不是同一個(gè),下面的代碼展示了這個(gè)結(jié)論:

  1. package jvm; 
  2. import java.io.IOException; 
  3. import java.io.InputStream; 
  4. public class ClassLoaderTest { 
  5.  public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, 
  6.  InstantiationException { 
  7.  ClassLoader myLoader = new ClassLoader() { 
  8.  @Override 
  9.  public Class<?> loadClass(String name) throws ClassNotFoundException { 
  10.  String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class"
  11.  InputStream inputStream = getClass().getResourceAsStream(fileName); 
  12.  if (inputStream == null) { 
  13.  return super.loadClass(name); 
  14.  } 
  15.  try { 
  16.  byte[] b = new byte[inputStream.available()]; 
  17.  inputStream.read(b); 
  18.  return defineClass(name, b, 0, b.length); 
  19.  } catch (IOException e) { 
  20.  throw new ClassNotFoundException(); 
  21.  } 
  22.  } 
  23.  }; 
  24.  Object obj = myLoader.loadClass("jvm.ClassLoaderTest").newInstance(); 
  25.  System.out.println(obj.getClass()); 
  26.  System.out.println(obj instanceof jvm.ClassLoaderTest); 
  27.  ClassLoaderTest classLoaderTest = new ClassLoaderTest(); 
  28.  System.out.println(classLoaderTest.getClass()); 
  29.  System.out.println(classLoaderTest instanceof jvm.ClassLoaderTest); 
  30.  } 

上述代碼的運(yùn)行結(jié)果是:

 

竟然還有人沒搞懂 JVM 類加載器?一文徹底明白

 

可以看出,代碼中使用自定義類加載器(myLoader)加載的jvm.ClassLoaderTest類和通過應(yīng)用程序類加載器加載的類不是同一個(gè)類。綜上,類加載器在JVM中的作用有:

  1. 將類的字節(jié)碼文件從JVM外部加載到內(nèi)存中
  2. 確定一個(gè)類的唯一性
  3. 提供隔離特性,為中間件開發(fā)者提供便利,例如Tomcat

 

責(zé)任編輯:武曉燕 來源: 今日頭條
相關(guān)推薦

2024-05-11 14:18:44

迭代器模式業(yè)務(wù)

2021-06-30 08:45:02

內(nèi)存管理面試

2022-06-07 10:13:22

前端沙箱對象

2020-03-18 14:00:47

MySQL分區(qū)數(shù)據(jù)庫

2020-12-07 06:19:50

監(jiān)控前端用戶

2021-07-08 10:08:03

DvaJS前端Dva

2019-08-27 14:46:59

ElasticSearES數(shù)據(jù)庫

2024-05-09 09:09:19

組合模式對象

2024-05-10 08:43:04

外觀模式接口系統(tǒng)

2024-05-13 10:45:25

中介模式面向?qū)ο?/a>數(shù)量

2019-11-06 17:30:57

cookiesessionWeb

2023-05-29 08:45:45

Java注解數(shù)據(jù)形式

2024-05-17 10:08:59

享元模式分類方式

2024-08-08 14:57:32

2022-04-11 10:56:43

線程安全

2024-05-15 17:41:37

備忘錄模式多線程

2023-04-12 08:38:44

函數(shù)參數(shù)Context

2021-08-05 06:54:05

觀察者訂閱設(shè)計(jì)

2023-11-23 06:50:08

括號

2020-12-18 09:36:01

JSONP跨域面試官
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號