关于ClassLoad的技术(动态装载)

王朝java/jsp·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

/*在运行时刻从文件中调入Class (ClassLoader 的使用)

本文介绍如何使用类库加载器ClassLoader 实现在运行时刻调入class

方法1

使用接口类,新调用的class是对它的具体实现

1) 写一个接口类 newface.class

2) 写接口文件实现 testfacea.class 更名为 testfacea.file 或其它文件名全可以

3) 主程序中调入文件到byte[]中,可以在本地文件调用,也可用网络无论如何只要能将

编译后的文件内容的类代码放到 byte[]当中就可以

4) 转换成一个Class并初始化

5) 实现拉口

实际上就是对一个接口类用调入的文件实现,当然可以用不同的文件进行不同的实现

也可以对一个文件进行加解密操作,

要注意的是对一个要调入的文件,一定要是一个已经存在的接口类的实现

这个有点EJB中的调用的中远程接口要在本地,而调入的文件就是EJBobject了

这种方法的的好处是要调入的class中的方法是可以说是已知的,相对来讲这种方法

简明易用

其它:

但是Java并不提供一个类似于类库卸载器(ClassUnloader) 的功能,能够把已经装载的模块从内存

里面清除掉。所以如果一个class已经调入,就放在了内存当中,并在程序信息表内记录本class已经

调入,如果再一次使用就从内存当中调入,如果实际文件更新也只能是程序下次运行时间再调入,而不

能在运行期间更新一个class,如果强行再一次调入就会

Linkage Error: duplicate class definition (重复定义class)

*/

package com;

//使用的主程序

public class testnewface {

public static void main(String[] args) throws java.lang.Exception

{

// 共用初使化参数,开始

Class testc;

Object testo;

cloader cl=new cloader();

// 共用初使化参数,结束

//方法 1 的例子代码,newface是本地接口类,newface.class本地已经存在 开始

testc=cl.load("testfacea.class","testfacea");

testo=testc.newInstance();

System.out.println("Instance has over!");

((newface)testo).out("方法1 第(1)种使用方法");

System.out.println("outsize="+((newface)testo).outsize("1111","aaaa"));

}

}

//接口

package com;

public interface newface {

public void out(String xx);

public int outsize(String x1,String x2);

}

//实现接口的类--编译后的.class文件放到D://

package com;

public class testfacea implements newface{

public void out(String xx) {

System.out.println(xx+" for testfacea ");

}

public int outsize(String x1,String x2) {

return x1.length()+x2.length();

}

}

//自己的ClassLoad类Loader

package com;

/*

要想自己完成从一个 byte[] 转换到一个Class 必须要 extends ClassLoader

因为ClassLoader中的方法defineClass是 protected 要使用只有 extends ClassLoader

*/

//import com.newface;

class cloader extends ClassLoader {

static int maxsize=10000;

public Class load(String namefile,String classname) throws java.lang.Exception

{

Class result=null;

try {

//进行判断这个class是否已经调入,已经有就直接返回,不然就调入

Class ctmp=this.findLoadedClass(classname);

System.out.println(ctmp.getName()+" is load");

return ctmp;

}

catch (Exception e) {

System.out.println("memory is null");

}

System.out.println("start..");

java.io.FileInputStream in=new java.io.FileInputStream("d://"+namefile);

byte[] classbyte=new byte[maxsize];

//实际应用时完全可以对一个文件进行加解密处理,只要保证使用defineClass时classbyte中

//已经解密后的内容就可以

int readsize;

readsize=in.read(classbyte);

System.out.println("读文件长:"+readsize);

in.close();

System.out.println("the classname:"+classname);

return defineClass("com."+classname,classbyte,0,readsize);

}

}

********************************************************************

注意:编译后的.class文件的包名,需要同我们的文件的是同一个包下的。

*********************************************************************

部署到Jboss 上还存在问题,即我们的类找不到!!

*********************************************************************

注:以上是我们根据网上某网友ClassLoad的示例进行的修改。大家共同交流。

*********************************************************************

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航