Java實現(xiàn)的自定義類加載器示例
本文實例講述了Java實現(xiàn)的自定義類加載器。分享給大家供大家參考,具體如下:
一 點睛
1 ClassLoader類有如下兩個關(guān)鍵方法:
loadClass(String name, boolean resolve):該方法為ClassLoader的入口點,根據(jù)指定的二進制名稱來加載類,系統(tǒng)就是調(diào)用ClassLoader的該方法來獲取指定類對應(yīng)的Class對象。
findClass(String name):根據(jù)二進制名稱來查找類。
如果需要實現(xiàn)自定義的ClassLoader,可以通過重寫以上兩個方法來實現(xiàn),當(dāng)然我們推薦重寫findClass()方法,而不是重寫loadClass()方法。
2 自定義類加載器常用功能
執(zhí)行代碼前自動驗證數(shù)字簽名。
根據(jù)用戶提供的密碼解密代碼,從而可以實現(xiàn)代碼混淆器來避免反編譯class文件。
根據(jù)用戶需求來動態(tài)地加載類。
根據(jù)應(yīng)用需求把其他數(shù)據(jù)以字節(jié)碼的形式加載到應(yīng)用中。
二 實戰(zhàn)
1 CompileClassLoader.java
import java.io.*;
import java.lang.reflect.*;
public class CompileClassLoader extends ClassLoader
{
// 讀取一個文件的內(nèi)容
private byte[] getBytes(String filename)
throws IOException
{
File file = new File(filename);
long len = file.length();
byte[] raw = new byte[(int)len];
try(
FileInputStream fin = new FileInputStream(file))
{
// 一次讀取class文件的全部二進制數(shù)據(jù)
int r = fin.read(raw);
if(r != len)
throw new IOException("無法讀取全部文件:"
+ r + " != " + len);
return raw;
}
}
// 定義編譯指定Java文件的方法
private boolean compile(String javaFile)
throws IOException
{
System.out.println("CompileClassLoader:正在編譯 "
+ javaFile + "...");
// 調(diào)用系統(tǒng)的javac命令
Process p = Runtime.getRuntime().exec("javac " + javaFile);
try
{
// 其他線程都等待這個線程完成
p.waitFor();
}
catch(InterruptedException ie)
{
System.out.println(ie);
}
// 獲取javac線程的退出值
int ret = p.exitValue();
// 返回編譯是否成功
return ret == 0;
}
// 重寫ClassLoader的findClass方法
protected Class<?> findClass(String name)
throws ClassNotFoundException
{
Class clazz = null;
// 將包路徑中的點(.)替換成斜線(/)。
String fileStub = name.replace("." , "/");
String javaFilename = fileStub + ".java";
String classFilename = fileStub + ".class";
File javaFile = new File(javaFilename);
File classFile = new File(classFilename);
// 當(dāng)指定Java源文件存在,且class文件不存在、或者Java源文件
// 的修改時間比class文件修改時間更晚,重新編譯
if(javaFile.exists() && (!classFile.exists()
|| javaFile.lastModified() > classFile.lastModified()))
{
try
{
// 如果編譯失敗,或者該Class文件不存在
if(!compile(javaFilename) || !classFile.exists())
{
throw new ClassNotFoundException(
"ClassNotFoundExcetpion:" + javaFilename);
}
}
catch (IOException ex)
{
ex.printStackTrace();
}
}
// 如果class文件存在,系統(tǒng)負(fù)責(zé)將該文件轉(zhuǎn)換成Class對象
if (classFile.exists())
{
try
{
// 將class文件的二進制數(shù)據(jù)讀入數(shù)組
byte[] raw = getBytes(classFilename);
// 調(diào)用ClassLoader的defineClass方法將二進制數(shù)據(jù)轉(zhuǎn)換成Class對象
clazz = defineClass(name,raw,0,raw.length);
}
catch(IOException ie)
{
ie.printStackTrace();
}
}
// 如果clazz為null,表明加載失敗,則拋出異常
if(clazz == null)
{
throw new ClassNotFoundException(name);
}
return clazz;
}
// 定義一個主方法
public static void main(String[] args) throws Exception
{
// 如果運行該程序時沒有參數(shù),即沒有目標(biāo)類
if (args.length < 1)
{
System.out.println("缺少目標(biāo)類,請按如下格式運行Java源文件:");
System.out.println("java CompileClassLoader ClassName");
}
// 第一個參數(shù)是需要運行的類
String progClass = args[0];
// 剩下的參數(shù)將作為運行目標(biāo)類時的參數(shù),
// 將這些參數(shù)復(fù)制到一個新數(shù)組中
String[] progArgs = new String[args.length-1];
System.arraycopy(args , 1 , progArgs
, 0 , progArgs.length);
CompileClassLoader ccl = new CompileClassLoader();
// 加載需要運行的類
Class<?> clazz = ccl.loadClass(progClass);
// 獲取需要運行的類的主方法
Method main = clazz.getMethod("main" , (new String[0]).getClass());
Object[] argsArray = {progArgs};
main.invoke(null,argsArray);
}
}
2 Hello.java
public class Hello
{
public static void main(String[] args)
{
for (String arg : args)
{
System.out.println("運行Hello的參數(shù):" + arg);
}
}
}
3 運行
E:\Java\瘋狂java講義\codes\18\18.2>java CompileClassLoader Hello 自定義加載器
CompileClassLoader:正在編譯 Hello.java...
運行Hello的參數(shù):自定義加載器
更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Java面向?qū)ο蟪绦蛟O(shè)計入門與進階教程》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對大家java程序設(shè)計有所幫助。
相關(guān)文章
IDEA:Git stash 暫存分支修改的實現(xiàn)代碼
這篇文章主要介紹了IDEA:Git stash 暫存分支修改的實現(xiàn)代碼,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-03-03
mybatis的映射xml中動態(tài)設(shè)置orderby方式
這篇文章主要介紹了mybatis的映射xml中動態(tài)設(shè)置orderby方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11
Spring boot JPA實現(xiàn)分頁和枚舉轉(zhuǎn)換代碼示例
這篇文章主要介紹了Spring boot JPA實現(xiàn)分頁和枚舉轉(zhuǎn)換代碼示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-09-09
零基礎(chǔ)寫Java知乎爬蟲之將抓取的內(nèi)容存儲到本地
上一回我們說到了如何把知乎的某些內(nèi)容爬取出來,那么這一回我們就說說怎么把這些內(nèi)容存儲到本地吧。2014-11-11

