[Java類加載器]Java中classLoader淺析.

本文爲在公司內部TD上寫的一篇小文, 主要講解java中classLoader基礎知識, 如今拿來這裏分享一下. java

1、問題c++

請在Eclipse中新建以下類,並運行它:app

1 package java.lang;
2 
3 public class Long { 4 public static void main(String[] args) { 5 System.out.println("Hi, i am here"); 6  } 7 }

你能猜到它的運行若是嗎? 不錯,就是以下這個樣子!spa

錯誤: 在類 java.lang.Long 中找不到 main 方法, 請將 main 方法定義爲:
public static void main(String[] args)
不然 JavaFX 應用程序類必須擴展javafx.application.Applicationcode

 





爲何呢,明明我在Long方法類中定義了main方法,爲何說main方法沒有定義呢?對象

本文將解決以上問題出現的緣由。blog

2、ClassLoader的做用繼承

咱們都知道java程序寫好之後是以.java(文本文件)的文件存在磁盤上,而後,咱們經過(bin/javac.exe)編譯命令把.java文件編譯成.class文件(字節碼文件),並存在磁盤上。可是程序要運行,首先必定要把.class文件加載到JVM內存中才能使用的,咱們所講的classLoader,就是負責把磁盤上的.class文件加載到JVM內存中,以下圖所示:

ip

你能夠認爲每個Class對象擁有磁盤上的那個.class字節碼內容,每個class對象都有一個getClassLoader()方法,獲得是誰把我從.class文件加載到內存中變成Class對象的。內存

3、ClassLoader層次結構

請執行以下程序:

 1 public class Test {
 2     public static void main(String[] args) { 3 ClassLoader classLoader = Test.class.getClassLoader(); 4  System.out.println(classLoader); 5 6 ClassLoader classLoader1 = classLoader.getParent(); 7  System.out.println(classLoader1); 8 9 ClassLoader classLoader2 = classLoader1.getParent(); 10  System.out.println(classLoader2); 11  } 12 }

它的輸出是:

sun.misc.Launcher$AppClassLoader@2a139a55
sun.misc.Launcher$ExtClassLoader@7852e922
null

獲得了 classLoader2就是null值了。這裏其實有三個類加載器:

(1): 根類加載器(null)

它是由本地代碼(c/c++)實現的,你根本拿不到他的引用,可是他實際存在,而且加載一些重要的類,它加載(%JAVA_HOME%\jre\lib),如rt.jar(runtime)、i18n.jar等,這些是Java的核心類。

(2): 擴展類加載器(ExtClassLoader)

雖然說能拿到,可是咱們在實踐中不多用到它,它主要加載擴展目錄下的jar包, %JAVA_HOME%\lib\ext

(3): 應用類加載器(AppClassLoader)

它主要加載咱們應用程序中的類,如Test,或者用到的第三方包,如jdbc驅動包等。

這裏的父類加載器與類中繼承概念要區分,它們在class定義上是沒有父子關係的。

4、Class加載時調用類加載器的順序

當一個類要被加載時,有一個啓動類加載器和實際類加載器的概念,這個概念請看以下分析

如上面的Test.class要進行加載時,它將會啓動應用類加載器進行加載Test類,可是這個應用類加載器不會真正去加載他,而是會調用看是否有父加載器,結果有,是擴展類加載器,擴展類加載器也不會直接去加載,它看本身是否有父加載器沒,結果它仍是有的,是根類加載器。

因此這個時候根類加載器就去加載這個類,可在%JAVA_HOME%\jre\lib下,它找不到com.wangmeng.Test這個類,因此他告訴他的子類加載器,我找不到,你去加載吧,子類擴展類加載器去%JAVA_HOME%\lib\ext去找,也找不着,它告訴它的子類加載器 AppClassLoader,我找不到這個類,你去加載吧,結果AppClassLoader找到了,就加到內存中,並生成Class對象。
這個時間時候啓動類加載器(應用類加載器)和實際類加載器(應用類加載器)是同一個.

這就是Java中著名的委託加載機制,看以下圖:

咱們再來看一下 java.lang.Long的加載,按上面分析,應該是由根類加載器加載獲得的,此時啓動類加載器是應用類加載器,但實際類加載器是根類加載器。

因此回到咱們最開始那個問題,沒有main方法是由於執行的根本不是咱們本身寫的類,執行的是java核心中的那個Long類,固然沒有main方法了。 這樣就防止咱們應用中寫的類覆蓋掉java核心類。

相關文章
相關標籤/搜索