ClassLoader一個常常出現又讓不少人望而卻步的詞,本文將試圖以最淺顯易懂的方式來說解 ClassLoader,但願能對不瞭解該機制的朋友起到一點點做用。
要深刻了解ClassLoader,首先就要知道ClassLoader是用來幹什麼的,顧名思義,它就是用來加載Class文件到JVM,以供程序使用的。咱們知道,java程序能夠動態加載類定義,而這個動態加載的機制就是經過ClassLoader來實現的,因此可想而知ClassLoader的重要性如何。
看到這裏,可能有的朋友會想到一個問題,那就是既然ClassLoader是用來加載類到JVM中的,那麼ClassLoader又是如何被加載呢?難道它不是java的類?
沒有錯,在這裏確實有一個ClassLoader不是用java語言所編寫的,而是JVM實現的一部分,這個ClassLoader就是bootstrap classloader(啓動類加載器),這個ClassLoader在JVM運行的時候加載java核心的API以知足java程序最基本的需求,其中就包括用戶定義的ClassLoader,這裏所謂的用戶定義是指經過java程序實現的ClassLoader,一個是ExtClassLoader,這個ClassLoader是用來加載java的擴展API的,也就是/lib/ext中的類,一個是AppClassLoader,這個ClassLoader是用來加載用戶機器上CLASSPATH設置目錄中的Class的,一般在沒有指定ClassLoader的狀況下,程序員自定義的類就由該ClassLoader進行加載。
當運行一個程序的時候,JVM啓動,運行bootstrap classloader,該ClassLoader加載java核心API(ExtClassLoader和AppClassLoader也在此時被加載),而後調用ExtClassLoader加載擴展API,最後AppClassLoader加載CLASSPATH目錄下定義的Class,這就是一個程序最基本的加載流程。
上面大概講解了一下ClassLoader的做用以及一個最基本的加載流程,接下來將講解一下ClassLoader加載的方式,這裏就不得不講一下ClassLoader在這裏使用了雙親委託模式進行類加載。
每個自定義ClassLoader都必須繼承ClassLoader這個抽象類,而每一個ClassLoader都會有一個parent ClassLoader,咱們能夠看一下ClassLoader這個抽象類中有一個getParent()方法,這個方法用來返回當前ClassLoader的parent,注意,這個parent不是指的被繼承的類,而是在實例化該ClassLoader時指定的一個ClassLoader,若是這個parent爲null,那麼就默認該ClassLoader的parent是bootstrap classloader,這個parent有什麼用呢?
咱們能夠考慮這樣一種狀況,假設咱們自定義了一個ClientDefClassLoader,咱們使用這個自定義的ClassLoader加載java.lang.String,那麼這裏String是否會被這個ClassLoader加載呢?事實上java.lang.String這個類並非被這個ClientDefClassLoader加載,而是由bootstrap classloader進行加載,爲何會這樣?實際上這就是雙親委託模式的緣由,由於在任何一個自定義ClassLoader加載一個類以前,它都會先委託它的父親ClassLoader進行加載,只有當父親ClassLoader沒法加載成功後,纔會由本身加載,在上面這個例子裏,由於java.lang.String是屬於java核心API的一個類,因此當使用ClientDefClassLoader加載它的時候,該ClassLoader會先委託它的父親ClassLoader進行加載,上面講過,當ClassLoader的parent爲null時,ClassLoader的parent就是bootstrap classloader,因此在ClassLoader的最頂層就是bootstrap classloader,所以最終委託到bootstrap classloader的時候,bootstrap classloader就會返回String的Class。
咱們來看一下ClassLoader中的一段源代碼:
java
protectedsynchronized Class loadClass(String name, boolean resolve) 程序員
throws ClassNotFoundException bootstrap
{ api
// 首先檢查該name指定的class是否有被加載緩存
Class c = findLoadedClass(name); 安全
if (c == null) { 網絡
try { app
if (parent != null) { jvm
//若是parent不爲null,則調用parent的loadClass進行加載ide
= parent.loadClass(name, false);
} else {
//parent爲null,則調用BootstrapClassLoader進行加載
c = findBootstrapClass0(name);
}
} catch (ClassNotFoundException e) {
//若是仍然沒法加載成功,則調用自身的findClass進行加載
c = findClass(name);
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
經過ClassLoader咱們能夠自定義類加載器,定製本身所須要的加載方式,例如從網絡加載,從其餘格式的文件加載等等均可以,其實ClassLoader還有不少地方沒有講到,例如ClassLoader內部的一些實現等等,原本但願可以講得簡單易懂一點,但是結果本身看回頭好像感受並不怎麼樣,鬱悶,看來本身的文筆仍是差太多了,但願可以給一些有須要的朋友一點幫助吧。
JVM在加載類的時候,都是經過ClassLoader的loadClass()方法來加載class的,loadClass(String name)方法:
使用的是雙親委託模式:
jvm啓動時,會啓動jre/rt.jar裏的類加載器:bootstrap classloader,用來加載java核心api;而後啓動擴展類加載器ExtClassLoader加載擴展類,並加載用戶程序加載器AppClassLoader,並指定ExtClassLoader爲他的父類;
當類被加載時,會先檢查在內存中是否已經被加載,若是是,則再也不加載,若是沒有,再由AppClassLoader來加載,先從jar包裏找,沒有再從classpath裏找;
若是自定義loader類,就會存在這命名空間的狀況,不一樣的加載器加載同一個類時,產生的實例實際上是不一樣的;
Java代碼
public Class<?> loadClass(String name) throws ClassNotFoundException {
return loadClass(name, false);
}
public Class<?> loadClass(String name) throws ClassNotFoundException {
return loadClass(name, false);
}
loadClass(String name)方法再調用loadClass(String name, boolean resolve)方法:
◆ name - 類的二進制名稱
◆ resolve - 若是該參數爲 true,則分析這個類
Java代碼
protectedsynchronized Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
// First, check if the class has already been loaded
//JVM 規範規定ClassLoader能夠在緩存保留它所加載的Class,若是一個Class已經被加載過,則直接從緩存中獲取
Class c = findLoadedClass(name);
if (c == null) {
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClass0(name);
}
} catch (ClassNotFoundException e) {
// If still not found, then invoke findClass in order
// to find the class.
c = findClass(name);
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
protectedsynchronized Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
// First, check if the class has already been loaded
//JVM 規範規定ClassLoader能夠在緩存保留它所加載的Class,若是一個Class已經被加載過,則直接從緩存中獲取
Class c = findLoadedClass(name);
if (c == null) {
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClass0(name);
}
} catch (ClassNotFoundException e) {
// If still not found, then invoke findClass in order
// to find the class.
c = findClass(name);
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
若是ClassLoader並無加載這個class,則調用findBootstrapClass0:
Java代碼
private Class findBootstrapClass0(String name)
throws ClassNotFoundException
{
check();
if (!checkName(name))
thrownew ClassNotFoundException(name);
return findBootstrapClass(name);
}
private Class findBootstrapClass0(String name)
throws ClassNotFoundException
{
check();
if (!checkName(name))
thrownew ClassNotFoundException(name);
return findBootstrapClass(name);
}
該方法會調用check()方法來判斷這個類是否已經初始化,而且經過checkName(name)來判斷由name指定的這個類是否存在最後調用findBootstrapClass(name):
Java代碼
privatenative Class findBootstrapClass(String name)
throws ClassNotFoundException;
privatenative Class findBootstrapClass(String name)
throws ClassNotFoundException;
而這個findBootstrapClass方法是一個native方法,這是咱們的root loader,這個載入方法並不是是由JAVA所寫,而是C++寫的,它會最終調用JVM中的原生findBootstrapClass方法來完成類的加載。
若是上面兩個都找不到,則使用findClass(name)來查找指定類名的Class:
Java代碼
protected Class<?> findClass(String name) throws ClassNotFoundException {
thrownew ClassNotFoundException(name);
}
protected Class<?> findClass(String name) throws ClassNotFoundException {
thrownew ClassNotFoundException(name);
}
JDK5.0中的說明:
使用指定的二進制名稱查找類。此方法應該被類加載器的實現重寫,該實現按照委託模型來加載類。在經過父類加載器檢查所請求的類後,此方法將被 loadClass 方法調用。默認實現拋出一個ClassNotFoundException。
因此,咱們在自定義類中,只須要重寫findClass()便可。
MyClassLoader類:
Java代碼
publicclass MyClassLoader extends ClassLoader {
private String fileName;
public MyClassLoader(String fileName) {
this.fileName = fileName;
}
protected Class<?> findClass(String className) throws ClassNotFoundException {
Class clazz = this.findLoadedClass(className);
if (null == clazz) {
try {
String classFile = getClassFile(className);
FileInputStream fis = new FileInputStream(classFile);
FileChannel fileC = fis.getChannel();
ByteArrayOutputStream baos = new
ByteArrayOutputStream();
WritableByteChannel outC = Channels.newChannel(baos);
ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
while (true) {
int i = fileC.read(buffer);
if (i == 0 || i == -1) {
break;
}
buffer.flip();
outC.write(buffer);
buffer.clear();
}
fis.close();
byte[] bytes = baos.toByteArray();
clazz = defineClass(className, bytes, 0, bytes.length);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
return clazz;
}
privatebyte[] loadClassBytes(String className) throws
ClassNotFoundException {
try {
String classFile = getClassFile(className);
FileInputStream fis = new FileInputStream(classFile);
FileChannel fileC = fis.getChannel();
ByteArrayOutputStream baos = new
ByteArrayOutputStream();
WritableByteChannel outC = Channels.newChannel(baos);
ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
while (true) {
int i = fileC.read(buffer);
if (i == 0 || i == -1) {
break;
}
buffer.flip();
outC.write(buffer);
buffer.clear();
}
fis.close();
return baos.toByteArray();
} catch (IOException fnfe) {
thrownew ClassNotFoundException(className);
}
}
private String getClassFile(String name) {
StringBuffer sb = new StringBuffer(fileName);
name = name.replace('.', File.separatorChar) + ".class";
sb.append(File.separator + name);
return sb.toString();
}
}
publicclass MyClassLoader extends ClassLoader {
private String fileName;
public MyClassLoader(String fileName) {
this.fileName = fileName;
}
protected Class<?> findClass(String className) throws ClassNotFoundException {
Class clazz = this.findLoadedClass(className);
if (null == clazz) {
try {
String classFile = getClassFile(className);
FileInputStream fis = new FileInputStream(classFile);
FileChannel fileC = fis.getChannel();
ByteArrayOutputStream baos = new
ByteArrayOutputStream();
WritableByteChannel outC = Channels.newChannel(baos);
ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
while (true) {
int i = fileC.read(buffer);
if (i == 0 || i == -1) {
break;
}
buffer.flip();
outC.write(buffer);
buffer.clear();
}
fis.close();
byte[] bytes = baos.toByteArray();
clazz = defineClass(className, bytes, 0, bytes.length);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
return clazz;
}
privatebyte[] loadClassBytes(String className) throws
ClassNotFoundException {
try {
String classFile = getClassFile(className);
FileInputStream fis = new FileInputStream(classFile);
FileChannel fileC = fis.getChannel();
ByteArrayOutputStream baos = new
ByteArrayOutputStream();
WritableByteChannel outC = Channels.newChannel(baos);
ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
while (true) {
int i = fileC.read(buffer);
if (i == 0 || i == -1) {
break;
}
buffer.flip();
outC.write(buffer);
buffer.clear();
}
fis.close();
return baos.toByteArray();
} catch (IOException fnfe) {
thrownew ClassNotFoundException(className);
}
}
private String getClassFile(String name) {
StringBuffer sb = new StringBuffer(fileName);
name = name.replace('.', File.separatorChar) + ".class";
sb.append(File.separator + name);
return sb.toString();
}
}
該類中經過調用defineClass(String name, byte[] b, int off, int len)方法來定義一個類:
Java代碼
protectedfinal Class<?> defineClass(String name, byte[] b, int off, int len)
throws ClassFormatError
{
return defineClass(name, b, off, len, null);
}
protectedfinal Class<?> defineClass(String name, byte[] b, int off, int len)
throws ClassFormatError
{
return defineClass(name, b, off, len, null);
}
注:MyClassLoader加載類時有一個侷限,必需指定.class文件,而不能指定.jar文件。該類中的大部分代碼是從網上搜索到的,是出自一牛人之筆,只是不知道原帖在哪,但願不會被隱藏。
MainClassLoader類:
Java代碼
publicclass MainClassLoader {
publicstaticvoid main(String[] args) {
try {
MyClassLoader tc = new MyClassLoader("F:\\OpenLib\\");
Class c = tc.findClass("Test");
c.newInstance();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
}
}
}
publicclass MainClassLoader {
publicstaticvoid main(String[] args) {
try {
MyClassLoader tc = new MyClassLoader("F:\\OpenLib\\");
Class c = tc.findClass("Test");
c.newInstance();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
}
}
}
最後是一個簡單的Test測試類:
Java代碼
publicclass Test
{
public Test() {
System.out.println("Test");
}
publicstaticvoid main(String[] args) {
System.out.println("Hello World");
}
}