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)負責(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-03mybatis的映射xml中動態(tài)設(shè)置orderby方式
這篇文章主要介紹了mybatis的映射xml中動態(tài)設(shè)置orderby方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11Spring boot JPA實現(xiàn)分頁和枚舉轉(zhuǎn)換代碼示例
這篇文章主要介紹了Spring boot JPA實現(xiàn)分頁和枚舉轉(zhuǎn)換代碼示例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-09-09零基礎(chǔ)寫Java知乎爬蟲之將抓取的內(nèi)容存儲到本地
上一回我們說到了如何把知乎的某些內(nèi)容爬取出來,那么這一回我們就說說怎么把這些內(nèi)容存儲到本地吧。2014-11-11