剛開始學(xué)習(xí)加載的時(shí)候,接觸到的是HeloWorld程序,當(dāng)時(shí)不知道為什么在public static void main(String [] args) 方法里寫了System.out.println("Hello World!") 就可以在控制臺(tái)打出“Hello World!”來(lái),確實(shí)的說(shuō),是什么東西隱蔽在后面執(zhí)行了我們寫的這段代碼,通過后來(lái)的學(xué)習(xí),知道了所有的class都是通過classloader來(lái)加載的。java規(guī)范這么說(shuō),Java的ClassLoader就是用來(lái)動(dòng)態(tài)裝載class的,ClassLoader對(duì)一個(gè)class只會(huì)裝載一次,JVM使用的ClassLoader一共有4種:

         啟動(dòng)類裝載器,標(biāo)準(zhǔn)擴(kuò)展類裝載器,類路徑裝載器網(wǎng)絡(luò)類裝載器
這4種ClassLoader的優(yōu)先級(jí)依次從高到低,使用所謂的“雙親委派模型”。確切地說(shuō),如果一個(gè)網(wǎng)絡(luò)類裝載器被請(qǐng)求裝載一個(gè)java.lang.Integer,它會(huì)首先把請(qǐng)求發(fā)送給上一級(jí)的類路徑裝載器,如果返回已裝載,則網(wǎng)絡(luò)類裝載器將不會(huì)裝載這個(gè)java.lang.Integer,如果上一級(jí)的類路徑裝載器返回未裝載,它才會(huì)裝載java.lang.Integer。

       再說(shuō)說(shuō)Package權(quán)限。Java語(yǔ)言規(guī)定,在同一個(gè)包中的class,如果沒有修飾符,默認(rèn)為Package權(quán)限,包內(nèi)的class都可以訪問。但是這還不夠準(zhǔn)確。確切的說(shuō),只有由同一個(gè)ClassLoader裝載的class才具有以上的Package權(quán)限。比如啟動(dòng)類裝載器裝載了java.lang.String,類路徑裝載器裝載了我們自己寫的java.lang.Test,它們不能互相訪問對(duì)方具有Package權(quán)限的方法。這樣就阻止了惡意代碼訪問核心類的Package權(quán)限方法。    
       現(xiàn)在來(lái)通過擴(kuò)展ClassLoader類實(shí)現(xiàn)一個(gè)自己的類裝載器,每個(gè)Class對(duì)象都有一個(gè)引用指向裝載他的ClassLoader,可以通過public ClassLoader getClassLoader()方法得到它。為了創(chuàng)建自己的類裝載器我們應(yīng)該擴(kuò)展ClassLoader類,這是一個(gè)抽象類。假設(shè)要從本地文件系統(tǒng)使用我們實(shí)現(xiàn)的類裝載器裝載一個(gè)類,創(chuàng)建一個(gè)FileClassLoader extends ClassLoader,需要覆蓋ClassLoader中的findClass(String name)方法,這個(gè)方法通過類的名字而得到一個(gè)Class對(duì)象。

package test.International;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
public class FileClassLoader extends ClassLoader {
    
public static final String drive = "d:/";
    
public static final String fileType = ".class";
    
public FileClassLoader() {
        super();
    }

    
public FileClassLoader(ClassLoader arg0) {
        super(arg0);
    }

    
public Class findClass(String name) {
        
byte[] data = loadClassData(name);
        
return defineClass(name, data, 0, data.length);
    }

    
public byte[] loadClassData(String name) {
        FileInputStream fis 
= null;
        
byte[] data = null;
        
try {
            fis 
= new FileInputStream(new File(drive + name + fileType));
            ByteArrayOutputStream baos 
= new ByteArrayOutputStream();
            
int ch = 0;
            
while ((ch = fis.read()) != -1{
                baos.write(ch);
            }

            data 
= baos.toByteArray();
        }
 catch (IOException e) {
            e.printStackTrace();
        }

        
return data;
    }

    
public static void main(String[] args) throws Exception {
        FileClassLoader loader 
= new FileClassLoader();
        Class objClass 
= loader.loadClass("HelloWorld"true);
        Object obj 
= objClass.newInstance();
        System.
out.println(objClass.getName());
        System.
out.println(objClass.getClassLoader());
    }

}

現(xiàn)在把HelloWorld.java放到默認(rèn)包下,然后編譯,把HelloWorld.class放到D:\盤根目錄下,運(yùn)行FileClassLoader類,控制臺(tái)輸出:

HelloWorld
sun.misc.Launcher$AppClassLoader@82ba41
可見,我們的確加載上了HelloWorld.class