初探Java類載入機制

來源:互聯網
上載者:User

標籤:應用程式   文章   java   如何   

初探Java類載入機制

 一、在jdk1.2以後,類載入是通過委託來完成的,這意味著如果 ClassLoader 不能找到類,它會請求父代 ClassLoader 來執行此項任務,所有 ClassLoaders 的根是系統 ClassLoader,它會以預設方式裝入類 -- 即,從本地檔案系統。今天我們就來探討一下在jvm中這些機制是怎樣啟動並執行。讓我們假設有一個class位元組碼檔案(比如Hello.class檔案),那麼在應用程式中,他是如何被載入進來,並形成一個類對象的呢?我們這篇文章的目的就是為瞭解釋這個問題。

 

  在java.lang包裡有個ClassLoader類,ClassLoader 的基本目標是對類的請求提供服務。當 JVM 需要使用類時,它根據名稱向 ClassLoader 請求這個類,然後 ClassLoader 試圖返回一個表示這個類的 Class 對象。通過覆蓋對應於這個過程不同階段的方法,可以建立定製的 ClassLoader。其中有個loadClass(String name, boolean resolve)方法,該方法為ClassLoader的進入點,在jdk1.2以後,loadClass方法將預設調用findClass方法,詳細內容可以參考API文檔,我們編寫的ClassLoader主要就是為了覆蓋以上兩個方法。回到我們剛才的問題,怎樣讀進位元組碼檔案,並把它構成一個類對象呢?在ClassLoader裡有個方法,Class defineClass(String name, byte[] b, int off, int len),答案就在這裡了,我們根據把class位元組碼檔案(如Hello.class)讀進一個位元組數組裡,byte[] b,並把它轉化為Class對象,而這些資料可以來源於檔案,網路等,神奇吧:)

 

defineClass管理 JVM 的許多複雜、神秘和倚賴於實現的方面 -- 它把位元組碼分析成運行時資料結構、校正有效性等等。不必擔心,您無需親自編寫它。事實上,即使您想要這麼做也不能覆蓋它,因為它已被標記成最終的。

 

  其他一些方法:

 

findSystemClass方法:從本地檔案系統裝入檔案。它在本地檔案系統中尋找類檔案,如果存在,就使用 defineClass 將原始位元組轉換成 Class 對象,以將該檔案轉換成類。

 

findClass方法:jdk1.2以後loadClass 的預設實現調用這個新方法。findClass 的用途包含您的 ClassLoader 的所有特殊代碼,而無需要複製其它代碼(例如,當專門的方法失敗時,調用系統 ClassLoader)。

 

getSystemClassLoader: 如果覆蓋 findClass 或 loadClass,getSystemClassLoader 使您能以實際 ClassLoader 對象來訪問系統 ClassLoader(而不是固定的從 findSystemClass 調用它)。

 

getParent:為了將類請求委託給父代 ClassLoader,這個新方法允許 ClassLoader 擷取它的父代 ClassLoader。當使用特殊方法,定製的 ClassLoader 不能找到類時,可以使用這種方法。

 

resolveClass: 可以不完全地(不帶解析)裝入類,也可以完全地(帶解析)裝入類。當編寫我們自己的 loadClass 時,可以調用 resolveClass,這取決於 loadClass 的 resolve 參數的值。

findLoadedClass:充當一個緩衝,當請求 loadClass 裝入類時,它調用該方法來查看 ClassLoader 是否已裝入這個類,這樣可以避免重新裝入已存在類所造成的麻煩。應首先調用該方法。

 

  二、工作流程:

 

1)調用 findLoadedClass(String) 來查看是否存在已裝入的類,如果沒有,那麼採用那種特殊的神奇方式來擷取原始位元組。

 

2)通過父類ClassLoader調用loadClass方法,如果父類ClassLoader是null,那麼按預設方式裝入類,即系統ClassLoader。

 

3)調用findClass(String)去尋找類並擷取類;

 

4)如果loadClass 的 resolve 參數的值為true,那麼調用 resolveClass 解析 Class 對象.

 

5)如果還沒有類,返回 ClassNotFoundException。

 

6)否則,將類返回給調用程式。

 

  三、一個實現了ClassLoader的例子:

 

/**

*CompilingClassLoader.java

*Copyright 2005-2-12

*/

import java.io.*;

 

public class CompilingClassLoader extends ClassLoader{

//讀取一個檔案的內容

private byte[] getBytes(String filename) throws IOException{

File file=new File(filename);

long len=file.length();

byte[] raw=new byte[(int)len];

 

FileInputStream fin=new FileInputStream(file);

 

int r=fin.read(raw);

if(r!=len) throw new IOException("Can‘t read all,"+r+"!="+len);

 

fin.close();

 

return raw;

}

 

private boolean compile(String javaFile) throws IOException{

System.out.println("CCL:Compiling "+javaFile+"...");

//調用系統的javac命令

Process p=Runtime.getRuntime().exec("javac "+javaFile);

try{

//其他線程都等待這個線程完成

p.wait


初探Java類載入機制

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.