JAVA運行時動態加載類


  想必大家在J2EE開發中一定會納悶Servelt的改變,伴隨的是Tomcat的重啟。JAVA是否能夠動態加載類呢?答案是肯定的。當然這不局限於J2EE,也可以做更多的拓展。
Let's Hack The Code:

Java Files List:
ClassLoaderTest/
        IC.java
        Test.java
ClassLoaderTest/1/
          IC.java
          C.java
ClassLoaderTest/2/
          IC.java
          C.java
// 定義接口
IC.java

public interface IC
{
  public void action();
}

  


//實現接口1
C.java 

public class C implements IC
{
  public void action(){
    System.out.println("Hi i am A class.");
  }
}

  


//實現接口2
C.java

public class C implements IC
{
  public void action(){  
    System.out.println("Hi i am A NEW! class.");
  }
}

  



//測試類
Test.java

import java.util.*;
import java.net.*;
import java.io.*;

public class Test implements Runnable
{
  public Test(){
    new Thread(this).start();
  }

  public void run(){
    System.out.println("do Test run()...");
    Scanner sc = new Scanner(System.in);
    while(true){
      System.out.print("\nPls enter the input.\n1. do one more time\n0.exit the program\nonly accepted '1' or '0'\n~>");
      String in = sc.next();
      if("1".equals(in)){
        System.out.println("**********************************");
        System.out.println("Do load().");
        load();
        System.out.println("\n\n");
      }else if("0".equals(in)){
        System.out.println("Bye.");
        System.exit(0);
      }else{
        continue;
      }
    }
  }

  private void load(){
    String jarName = "C.jar";
    try{
      File file = new File(jarName);
      URL url = file.toURL();
      URLClassLoader loader = new URLClassLoader(new URL[]{url});
      Class aClass = loader.loadClass("C");

      IC ic = (IC)aClass.newInstance();
      ic.action();
    }catch (Exception e){
      e.printStackTrace();
    }
  }

  public static void main(String[] args){
    new Test();
  }
}

  

測試:
這里沒有用到包,寫完以上代碼要這樣測試

1.
在此目錄編譯C.java
ClassLoaderTest/1/
          IC.java
          C.java

javac C.java
把C.class封裝到jar中
jar cvf C1.jar C.class

2.
在此目錄編譯C.java
ClassLoaderTest/2/
          IC.java
          C.java

javac C.java
把C.class封裝到jar中
jar cvf C1.jar C.class


3.
編譯Test.java
ClassLoaderTest/
        IC.java
        Test.java

javac Test.java


4. 把C1.jar拷貝到ClassLoaderTest/目錄並改名為C.jar
cp C1.jar ../C.jar
執行java Test.
此時輸出為:

Pls enter the input.
1. do one more time
0.exit the program
only accepted '1' or '0'
~>1 
**********************************
Do load().
Hi i am A class.

  


現在用C2.jar替換C.jar

cp C2.jar ../C.jar

輸入1繼續執行,此時輸出應為:

Pls enter the input.
1. do one more time
0.exit the program
only accepted '1' or '0'
~>1 
**********************************
Do load().
Hi i am A NEW! class.

  

結論:
通過以上方式,就可以實現運行時動態加載類。如果你只是想實現運行時動態加載類,只需把上面的代碼做相應修改即可。
--------------------------------------------------------------------------------
更深入的研究

用Class.forName()替代。
在Test.java中的load()方法替換成

private void load(){
  try{
    Class aClass = Class.forName("C");
    IC ic = (IC)aClass.newInstance();
    ic.action();
  }catch (Exception e){
    e.printStackTrace();
  }
}

1.
在此目錄編譯C.java
ClassLoaderTest/1/
          IC.java
          C.java

javac C.java

2.
在此目錄編譯C.java
ClassLoaderTest/2/
          IC.java
          C.java

javac C.java

3.
編譯Test.java
ClassLoaderTest/
        IC.java
        Test.java

4. 把../1/C.class拷貝到ClassLoaderTest/中

cp C.class ../

執行java Test.
此時輸出為:

Pls enter the input.
1. do one more time
0.exit the program
only accepted '1' or '0'
~>1 
**********************************
Do load().
Hi i am A class.

再把../2/C.class拷貝到ClassLoaderTest/中
cp C.class ../
執行java Test.
此時輸出為:

Pls enter the input.
1. do one more time
0.exit the program
only accepted '1' or '0'
~>1 
**********************************
Do load().
Hi i am A class.

很驚訝的發現,同樣是調用了Class.newInstance()方法來獲取一個新的對象。但JVM並沒有載入我們新修改的.class文件。所以Class.forName()不能達到我們所期望的結果。


更多測試
速度比較,這里采用三種不同的方式測試。
1.通過URLClassLoader從jar文件中加載類並創建實例
2.通過Class.forName()加載類並創建實例
3.通過普通方式new創建實例

把A.java中的action()改為空方法。並且在幾個Test.java的run()方法中,作如下修改。

int MAX = 50000;
long start = System.currentTimeMillis();
for(int i=0;i<MAX;i++)
  load();
long end = System.currentTimeMillis();
System.out.println("time:"+ (end-start) );

 

MAX=5萬次
輸出
1.time:15795
2.time:39
3.time:20
MAX=10萬次
輸出
1.time:29735
2.time:24
3.time:19

這兩組測試中,同等條件下多次測試1/2所用時間都差不多,而3在執行多次后竟然時間趨於0,這也不排除是我機器原因。令人感到很有趣的是Class.forName()與普通的new在首次執行創建實例時,耗時幾乎所一致的,所以這也可以判斷,這兩種方式在首次創建實例時都是從磁盤中把.class文件載入再創建實例。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM