溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java中怎么自定義一個(gè)類(lèi)加載器

發(fā)布時(shí)間:2021-06-25 16:42:51 來(lái)源:億速云 閱讀:155 作者:Leah 欄目:編程語(yǔ)言

Java中怎么自定義一個(gè)類(lèi)加載器,相信很多沒(méi)有經(jīng)驗(yàn)的人對(duì)此束手無(wú)策,為此本文總結(jié)了問(wèn)題出現(xiàn)的原因和解決方法,通過(guò)這篇文章希望你能解決這個(gè)問(wèn)題。

一 點(diǎn)睛

1 ClassLoader類(lèi)有如下兩個(gè)關(guān)鍵方法:

loadClass(String name, boolean resolve):該方法為ClassLoader的入口點(diǎn),根據(jù)指定的二進(jìn)制名稱(chēng)來(lái)加載類(lèi),系統(tǒng)就是調(diào)用ClassLoader的該方法來(lái)獲取指定類(lèi)對(duì)應(yīng)的Class對(duì)象。

findClass(String name):根據(jù)二進(jìn)制名稱(chēng)來(lái)查找類(lèi)。

如果需要實(shí)現(xiàn)自定義的ClassLoader,可以通過(guò)重寫(xiě)以上兩個(gè)方法來(lái)實(shí)現(xiàn),當(dāng)然我們推薦重寫(xiě)findClass()方法,而不是重寫(xiě)loadClass()方法。

2 自定義類(lèi)加載器常用功能

執(zhí)行代碼前自動(dòng)驗(yàn)證數(shù)字簽名。

根據(jù)用戶提供的密碼解密代碼,從而可以實(shí)現(xiàn)代碼混淆器來(lái)避免反編譯class文件。

根據(jù)用戶需求來(lái)動(dòng)態(tài)地加載類(lèi)。

根據(jù)應(yīng)用需求把其他數(shù)據(jù)以字節(jié)碼的形式加載到應(yīng)用中。

二 實(shí)戰(zhàn)

1 CompileClassLoader.java

import java.io.*;import java.lang.reflect.*;public class CompileClassLoader extends ClassLoader{  // 讀取一個(gè)文件的內(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文件的全部二進(jìn)制數(shù)據(jù)     int r = fin.read(raw);     if(r != len)      throw new IOException("無(wú)法讀取全部文件:"         + 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   {     // 其他線程都等待這個(gè)線程完成     p.waitFor();   }   catch(InterruptedException ie)   {     System.out.println(ie);   }   // 獲取javac線程的退出值   int ret = p.exitValue();   // 返回編譯是否成功   return ret == 0;  }  // 重寫(xiě)ClassLoader的findClass方法  protected Class<?> findClass(String name)     throws ClassNotFoundException  {   Class clazz = null;   // 將包路徑中的點(diǎn)(.)替換成斜線(/)。   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源文件   // 的修改時(shí)間比class文件修改時(shí)間更晚,重新編譯   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對(duì)象   if (classFile.exists())   {     try     {      // 將class文件的二進(jìn)制數(shù)據(jù)讀入數(shù)組      byte[] raw = getBytes(classFilename);      // 調(diào)用ClassLoader的defineClass方法將二進(jìn)制數(shù)據(jù)轉(zhuǎn)換成Class對(duì)象      clazz = defineClass(name,raw,0,raw.length);     }     catch(IOException ie)     {      ie.printStackTrace();     }   }   // 如果clazz為null,表明加載失敗,則拋出異常   if(clazz == null)   {     throw new ClassNotFoundException(name);   }   return clazz;  }  // 定義一個(gè)主方法  public static void main(String[] args) throws Exception  {   // 如果運(yùn)行該程序時(shí)沒(méi)有參數(shù),即沒(méi)有目標(biāo)類(lèi)   if (args.length < 1)   {     System.out.println("缺少目標(biāo)類(lèi),請(qǐng)按如下格式運(yùn)行Java源文件:");     System.out.println("java CompileClassLoader ClassName");   }   // 第一個(gè)參數(shù)是需要運(yùn)行的類(lèi)   String progClass = args[0];   // 剩下的參數(shù)將作為運(yùn)行目標(biāo)類(lèi)時(shí)的參數(shù),   // 將這些參數(shù)復(fù)制到一個(gè)新數(shù)組中   String[] progArgs = new String[args.length-1];   System.arraycopy(args , 1 , progArgs      , 0 , progArgs.length);   CompileClassLoader ccl = new CompileClassLoader();   // 加載需要運(yùn)行的類(lèi)   Class<?> clazz = ccl.loadClass(progClass);   // 獲取需要運(yùn)行的類(lèi)的主方法   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("運(yùn)行Hello的參數(shù):" + arg);   }  }}

3 運(yùn)行

E:\Java\瘋狂java講義\codes\18\18.2>java CompileClassLoader Hello 

看完上述內(nèi)容,你們掌握J(rèn)ava中怎么自定義一個(gè)類(lèi)加載器的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI