主要是爲了安全,避免用戶惡意加載破壞JVM
正常運行的字節碼文件,好比說加載一個本身寫的java.util.HashMap.class
。這樣就有可能形成包衝突問題。java
jdk
中rt.jar
的字節碼文件jdk
中/jre/lib/ext
文件夾下的字節碼文件classPath
下的字節碼文件一、ClassLoader.loadClass()
git
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); // 若是這個Class對象尚未被加載,下面就準備加載 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 // from the non-null parent class loader } // 若是父類加載器也沒有加載這個Class對象,就由本身來加載 if (c == null) { // If still not found, then invoke findClass in order // to find the class. 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; } }
雙親委派加載模型僅僅是一個約定,後面實現類加載器時,是能夠不遵照這個約定。ClassLoader
是在JDK1.0
的時候就設計好的,而雙親委派加載模型在JDK1.2
引入的。因此,有些機制是沒有遵照這個約定的。好比:Service Provider Interface
機制的JDBC
就沒有遵照這個約定。github
一、爲何JDBC
沒法遵照這個約定?
JDBC
是SPI
機制的一個例子,JDK
定義了java.sql.Connection
核心接口,後續MySQL
、Oracle
爲其提供實現類。在運行中是經過java.sql.DriverManager
來獲取指定實現類的實例。這裏須要明白三個問題:sql
java.sql.DriverManager
是在rt.jar
中,由核心類加載器加載的;Collection
的實現類都是在classpath
中;也就是說當JVM
在java.sql.DriverManager
類的getConnection()
方法中獲取Collection
實現類的字節碼時,當前類的定義類加載器是啓動類加載器,而按照約定啓動類加載器是不容許加載classpath
下的字節碼。因此,JDBC
就沒法遵照這個約定。安全
二、JDBC
是如何解決上面的問題的?
爲了解決這個,java
在線程中放入一個類加載器Thread.currentThread().getContextClassLoader()
;而這個類加載器能夠是隨意的。好比你想加載classpath
包下的字節碼文件,只須要設置當前線程的類加載器爲應用程序類加載器便可。網絡
JVM
類加載過程JVM
本質的工做就是讀取字節碼文件、執行字節碼文件中的指令。其中JVM
將讀取字節碼文件的過程稱爲JVM
類加載過程。app
JVM
讀取的字節碼文件將放在方法區裏;jvm
JVM
類加載機制分爲五個部分:加載、驗證、準備、解析、初始化。以下圖所示:
ide
Loading
:加載這一步是將JVM
外的字節碼文件加載到JVM
內部方法區中的Class
對象。源碼分析
JVM
能夠經過幾種方式來加載外部的字節碼文件?因爲雙親委派加載模型的存在,一個Class
對象的初始類加載器initiating class loader
和定義類加載器defining class loader
有可能不是同一個。
JVM
加載這個字節碼文件defineClass
方法,將字節碼轉換成Class
對象java
在判斷instanceof
時,只有類名、defining class loader
都相等,才表示是同一個類的實例。
Class.getClassLoader()
獲得的是定義類加載器
一、驗證使用不一樣ClassLoader
加載字節碼文件
// 這種方法是不遵照雙親委派加載模型的約定 public class ClassLoaderLoading { public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { // 這個Class對象是由當前方法的類加載器加載 Class c1 = MiniJVM.class; Class c2 = new MyClassLoader().loadClass("com.github.hcsp.MiniJVM"); // 使用c2建立一個MiniJVM實例 Object o = c2.getConstructor().newInstance(); System.out.println(o instanceof MiniJVM); MiniJVM demo = (MiniJVM) o; } private static class MyClassLoader extends ClassLoader { @Override public Class<?> loadClass(String name) throws ClassNotFoundException { if (name.contains("MiniJVM")) { try { byte[] bytes = Files.readAllBytes(new File("target/classes/com/github/hcsp/MiniJVM.class").toPath()); return defineClass(name, bytes, 0, bytes.length); } catch (IOException e) { throw new RuntimeException(e); } } else { return super.loadClass(name); } } } }
二、實現一個遵照雙親委派加載模型的類加載器
public class ClassLoaderLoading { public static void main(String[] args) throws ClassNotFoundException { Class c1 = MiniJVM.class; Class c2 = new MyClassLoader(ClassLoader.getSystemClassLoader()).loadClass("com.github.hcsp.MiniJVM"); System.out.println("c2 = " + c2); } private static class MyClassLoader extends ClassLoader { public MyClassLoader(ClassLoader systemClassLoader) { super(systemClassLoader); } @Override public Class<?> loadClass(String name) throws ClassNotFoundException { // 加載你想讓這個類加載器加載的字節碼文件 if (name.contains("MiniJVM")) { try { byte[] bytes = Files.readAllBytes(new File("target/classes/com/github/hcsp/MiniJVM.class").toPath()); return defineClass(name, bytes, 0, bytes.length); } catch (IOException e) { throw new RuntimeException(e); } } else { // 其餘的字節碼文件交由父類加載器加載 return super.loadClass(name); } } } }
Linking
:連接當一個.java
文件編譯成.class
文件時,裏面含有一個符號引用,好比/java/utils/HashMap
。Linking
是指將這符號引用與具體的class
對象連接起來。
每一個字節碼結構都有一個運行時常量池,它會存儲每一個符號引用和所對應的具體對象,以此實現連接。
Verification
:驗證字節碼的正確性Preparation
:爲static
成員賦默認初始值Resolution
:解析當前字節碼裏包含的其餘符號引用Initializing
執行初始化方法。好比下面的四個虛擬機指令:new
、getstatic
、putstatic
、invokestatic