JVM(四)JVM的雙親委派模型

一、兩種不一樣的類加載器

  從JAVA虛擬機的角度來說,只存在兩種不一樣的類加載器:一種是啓動類加載器(Bootstrap ClassLoader),這個類加載器使用C++語言實現,是虛擬機自身的一部分;另外一種就是全部其餘的類加載器,這些加載器都由Java語言實現,獨立於虛擬機外部,而且全都繼承自抽象類java,lang.ClassLoader。java

二、類加載器雙親委派模型圖

  

三、對4個加載器的理解

  (1)啓動類加載器:這個類加載器負責將存放在<JAVA_HOME>\lib 目錄下的,例如rt.jar包。this

  (2)擴展類加載器:這個類加載器由sun.misc.Launcher$ExtClassLoader實現。它負責加載<JAVA_HOME>\lib\ext 目錄下的。或者被java.ext.dirs系統變量所指定的路徑中的全部類庫,開發者可用直接使用擴展類加載器。spa

  (3)應用程序類加載器:這個類加載器由sun.misc.Launcher$AppClassLoader實現。因爲這個類加載器是ClassLoader中的getSystemClassLoader()方法的返回值,全部通常也稱它爲系統類加載器。它負責加載用戶路徑(ClassPath)上所指定的類庫,開發者可用直接使用這個類加載器,若是應用程序中沒有自定義過本身的類加載器,通常狀況下這個就是程序中默認的類加載器。code

  (4)自定義類加載器:就是本身自定義的類加載器。blog

四、雙親委派模型的實現

protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            // 首先,檢查請求的類是否已經被加載過
            Class c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // 若是父類加載器拋出ClassNotFoundException 
                    // from the non-null parent class loader
                    // 說明父類加載器沒法完成加載請求
                }

                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    // 在父類加載器沒法加載的時候,再調用自己的findClass方法進行類加載 
                    long t1 = System.nanoTime();
                    c = findClass(name);

                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }   

  注意:即便自定義了本身的類加載器,強行用defineClass()方法區加載一個以"java.lang"開頭的類也不會成功,若是嘗試這樣的話,將會收到一個由虛擬機本身拋出的"java.lang.ScurityException:Prohibitted package name:java.lang"異常。繼承

相關文章
相關標籤/搜索