重写ClassLoader,动态加载Class文件问题

ustbsjl 2008-07-27 11:28:00
重写ClassLoader,动态加载Class文件问题没有问题,但是在类型转换的时候出错,不知道该如何解决?请高手指点迷津!
...全文
1079 29 打赏 收藏 转发到动态 举报
写回复
用AI写文章
29 条回复
切换为时间正序
请发表友善的回复…
发表回复
shijiyu00 2008-10-22
  • 打赏
  • 举报
回复
你这个方法好像不能加载带有参数的类?
interpb 2008-07-28
  • 打赏
  • 举报
回复
最好还是用接口的方式吧 这样灵活一点 也高效一点
interpb 2008-07-28
  • 打赏
  • 举报
回复

com.umpay.test.HelloWorld helloWorld = (com.umpay.test.HelloWorld) obj;
helloWorld.sayHello();
改成



Method mm = obj .getClass().getMethod("sayHello", null);
mm.invoke(obj , null);
ustbsjl 2008-07-28
  • 打赏
  • 举报
回复
请再说一下反射机制怎么调用,谢谢!
冰思雨 2008-07-28
  • 打赏
  • 举报
回复
Object obj = classLoader.loadClass("com.umpay.test.HelloWorld", true).newInstance();//这段代码是使用classLoader对象
//加载com.umpay.test.HelloWorld ,然后,再实例化一个对象.但是这个对象,是由classLoader对象加载的类来实例化的.
com.umpay.test.HelloWorld helloWorld = (com.umpay.test.HelloWorld) obj;//这段代码,是要将obj对象进行强制类型转换.
//但是,所要强之类型转换的类,并不是classLoader加载的,虽然,两个类都是com.umpay.test.HelloWorld,
//但是,他们不是同一个ClassLoader加载到虚拟机的,所以,虚拟机会认为这是两个毫无关系的类,强制类型转换会出错.
/*
解决这个问题,有两个方法:一种是使用反射调用sayHello方法,另一种是,声明一个接口类,让com.umpay.test.HelloWorld实现接口类的sayHello方法.
这样,classLoader在加载com.umpay.test.HelloWorld类的时候,com.umpay.test.HelloWorld的父类(接口类)一般情况下,会被虚拟机的ClassLoad加载进来,这样,在做强制类型转换的时候,就是同一个接口类了.
*/



ustbsjl 2008-07-28
  • 打赏
  • 举报
回复

/*
* @(#)Main.java 2008-7-26
*
* Copyright 2008 UMPay, Inc. All rights reserved.
* Use is subject to license terms.
*/
package com.umpay.test;

/**
* 描述:
*
* @version 1.0
* @author 沈建林
* @since 2008-7-26
*/
public class Main {

/**
* @param args
*/
public static void main(String[] args) throws Exception {
// TODO Auto-generated method stub
while (true) {
try {
BytecodeClassLoader classLoader = new BytecodeClassLoader();
try {
Object obj = classLoader.loadClass(
"com.umpay.test.HelloWorld", true).newInstance();
System.out.println(obj);

com.umpay.test.HelloWorld helloWorld = (com.umpay.test.HelloWorld) obj;
helloWorld.sayHello();
} catch (InstantiationException inse) {
// TODO Auto-generated catch block
inse.printStackTrace();
} catch (IllegalAccessException illee) {
// TODO Auto-generated catch block
illee.printStackTrace();
} catch (ClassNotFoundException cnfe) {
// TODO Auto-generated catch block
cnfe.printStackTrace();
}
} catch (Exception e) {
e.printStackTrace();
}

Thread.sleep(3 * 1000);

if (false) {
break;
}
}

if (true) {
return;
}
}
}

ustbsjl 2008-07-28
  • 打赏
  • 举报
回复

/*
* @(#)HelloWorld.java 2008-7-26
*
* Copyright 2008 UMPay, Inc. All rights reserved.
* Use is subject to license terms.
*/
package com.umpay.test;

/**
* 描述:
*
* @version 1.0
* @author 沈建林
* @since 2008-7-26
*/
public class HelloWorld {

/**
* @param args
*/
public void sayHello() {
// TODO Auto-generated method stub
System.out.println("Hello, Java! Hello, Java!");
}

}

ustbsjl 2008-07-28
  • 打赏
  • 举报
回复

package com.umpay.test;

import java.io.*;

public class CompilingClassLoader extends ClassLoader {
// 读入文件,将它存为byte数组
private byte[] getBytes(String filename) throws IOException {

File file = new File(filename);
long len = file.length();

byte raw[] = new byte[(int) len];

FileInputStream fin = new FileInputStream(file);

int r = fin.read(raw);
if (r != len)
throw new IOException("Can't read all, " + r + " != "
+ len);

fin.close();

return raw;
}

// 以输入参数为文件名编译java文件,成功返回true,不成功返回false
private boolean compile(String javaFile) throws IOException {

System.out.println("CCL: Compiling " + javaFile + "...");

Process p = Runtime.getRuntime().exec("javac " + javaFile);

try {
p.waitFor();
} catch (InterruptedException ie) {
System.out.println(ie);
}

// 取得编译是否成功的参数
int ret = p.exitValue();

// ret==0表示编译成功
return ret == 0;
}

// 核心程序,查找并载入java文件,看其是否需要编译
// 如需要,则编译后将其实例化,否则直接实例化
public Class loadClass(String name, boolean resolve)
throws ClassNotFoundException {

Class clas = findLoadedClass(name);

// System.out.println( "findLoadedClass: "+clas );

// 生成文件路径
// 例如: java.lang.Object => java/lang/Object

String fileStub = name.replace('.', '/');

String javaFilename = "src/" + fileStub + ".java";
String classFilename = "classes/" + fileStub + ".class";

File javaFile = new File(javaFilename);
File classFile = new File(classFilename);

System.out.println(javaFile.getAbsolutePath());
System.out.println(classFile.getAbsolutePath());

// System.out.println( "j "+javaFile.lastModified()+" c "+
// classFile.lastModified() );

// 根据文件日期,判断是否要重编译

if (javaFile.exists()
&& (!classFile.exists() || javaFile.lastModified() > classFile
.lastModified())) {

try {
// 编译失败,或类文件不存在
if (!compile(javaFilename) || !classFile.exists()) {
throw new ClassNotFoundException("Compile failed: "
+ javaFilename);
}
} catch (IOException ie) {

throw new ClassNotFoundException(ie.toString());
}
}

// 在编译成功,或不需要编译的情况下,将class文件载入到byte数组
try {

byte raw[] = getBytes(classFilename);
// 将byte数组转化为Class类型实例
clas = defineClass(name, raw, 0, raw.length);
} catch (IOException ie) {
}

// System.out.println( "defineClass: "+clas );
// 如果clas==null,按系统默认方式载入class
if (clas == null) {
clas = findSystemClass(name);
}

// System.out.println( "findSystemClass: "+clas );

if (resolve && clas != null)
resolveClass(clas);

// 如果仍然不能载入,则抛出错误
if (clas == null)
throw new ClassNotFoundException(name);

// 返回载入的class实例
return clas;
}
}

ustbsjl 2008-07-28
  • 打赏
  • 举报
回复
谢谢大家的帮助,结贴!
胡矣 2008-07-28
  • 打赏
  • 举报
回复
如果
BytecodeClassLoader classLoader = new BytecodeClassLoader();
改为
CompilingClassLoader classLoader = new CompilingClassLoader();
的话
不知道是不是LZ笔误
还是另有BytecodeClassLoader 类?
胡矣 2008-07-28
  • 打赏
  • 举报
回复
我用LZ写的程序是可以执行的
而且我打印出他们俩的classLoader都是
sun.misc.Launcher$AppClassLoader@82ba41
sun.misc.Launcher$AppClassLoader@82ba41
penghao122 2008-07-28
  • 打赏
  • 举报
回复
学习下
interpb 2008-07-28
  • 打赏
  • 举报
回复
[Quote=引用 22 楼 ustbsjl 的回复:]
使用接口确实可以解决问题,但是现在还有一个问题,在更新文件的时候如果文件传了一半,这个时候去加载的时候会报错误,这个问题怎么解决呢?
[/Quote]

这个就要你自己定义纠错机制了

设置一些标志位啊 之类
大真 2008-07-28
  • 打赏
  • 举报
回复
理不解
ustbsjl 2008-07-28
  • 打赏
  • 举报
回复
使用接口确实可以解决问题,但是现在还有一个问题,在更新文件的时候如果文件传了一半,这个时候去加载的时候会报错误,这个问题怎么解决呢?
老班长涛哥 2008-07-28
  • 打赏
  • 举报
回复
认真的看了哈程序,学习了哈!~~
ldy214 2008-07-28
  • 打赏
  • 举报
回复
classLoader不熟悉
interpb 2008-07-28
  • 打赏
  • 举报
回复
我的代码就是一个简单的反射调用

与动态代理没有任何关系 也不需要实现InvocationHandler接口



interpb 2008-07-28
  • 打赏
  • 举报
回复
[Quote=引用 17 楼 luoxiaohu 的回复:]
楼上的.你这个应该不行把 这个是针对动态代理实现. 没有继承自InvocationHandler接口
却直接调用它的方法
[/Quote]

为什么自己不试试呢
luoxiaohu 2008-07-28
  • 打赏
  • 举报
回复
楼上的.你这个应该不行把 这个是针对动态代理实现. 没有继承自InvocationHandler接口
却直接调用它的方法
加载更多回复(7)
主要特性Java 语言是简单的:Java 语言的语法与 C 语言和 C++ 语言很接近,使得大多数程序员很容易学习和使用。另一方面,Java 丢弃了 C++ 中很少使用的、很难理解的、令人迷惑的那些特性,如操作符重载、多继承、自动的强制类型转换。特别地,Java 语言不使用指针,而是引用。并提供了自动分配和回收内存空间,使得程序员不必为内存管理而担忧。Java 语言是面向对象的:Java 语言提供类、接口和继承等面向对象的特性,为了简单起见,只支持类之间的单继承,但支持接口之间的多继承,并支持类与接口之间的实现机制(关键字为 implements)。Java 语言全面支持动态绑定,而 C++语言只对虚函数使用动态绑定。总之,Java语言是一个纯的面向对象程序设计语言。Java语言是分布式的:Java 语言支持 Internet 应用的开发,在基本的 Java 应用编程接口中有一个网络应用编程接口(java net),它提供了用于网络应用编程的类库,包括 URL、URLConnection、Socket、ServerSocket 等。Java 的 RMI(远程方法激活)机制也是开发分布式应用的重要手段。Java 语言是健壮的:Java 的强类型机制、异常处理、垃圾的自动收集等是 Java 程序健壮性的重要保证。对指针的丢弃是 Java 的明智选择。Java 的安全检查机制使得 Java 更具健壮性。Java语言是安全的:Java通常被用在网络环境中,为此,Java 提供了一个安全机制以防恶意代码的攻击。除了Java 语言具有的许多安全特性以外,Java 对通过网络下载的类具有一个安全防范机制(类 ClassLoader),如分配不同的名字空间以防替代本地的同名类、字节代码检查,并提供安全管理机制(类 SecurityManager)让 Java 应用设置安全哨兵。Java 语言是体系结构中立的:Java 程序(后缀为 java 的文件)在 Java 平台上被编译为体系结构中立的字节码格式(后缀为 class文件),然后可以在实现这个 Java 平台的任何系统中运行。这种途径适合于异构的网络环境和软件的分发。Java 语言是可移植的:这种可移植性来源于体系结构中立性,另外,Java 还严格规定了各个基本数据类型的长度。Java 系统本身也具有很强的可移植性,Java 编译器是用 Java 实现的,Java 的运行环境是用 ANSI C 实现的。Java 语言是解释型的:如前所述,Java 程序在 Java 平台上被编译为字节码格式,然后可以在实现这个 Java 平台的任何系统中运行。在运行时,Java 平台中的 Java 解释器对这些字节码进行解释执行,执行过程中需要的类在联接阶段被载入到运行环境中。Java 是高性能的:与那些解释型的高级脚本语言相比,Java 的确是高性能的。事实上,Java 的运行速度随着 JIT(Just-In-Time)编译器技术的发展越来越接近于 C++。Java 语言是多线程的:在 Java 语言中,线程是一种特殊的对象,它必须由 Thread 类或其子(孙)类来创建。通常有两种方法来创建线程:其一,使用型构为 Thread(Runnable) 的构造子类将一个实现了 Runnable 接口的对象包装成一个线程,其二,从 Thread 类派生出子类并重写 run 方法,使用该子类创建的对象即为线程。值得注意的是 Thread 类已经实现了 Runnable 接口,因此,任何一个线程均有它的 run 方法,而 run 方法中包含了线程所要运行的代码。线程的活动由一组方法来控制。Java 语言支持多个线程的同时执行,并提供多线程之间的同步机制(关键字为 synchronized)。Java 语言是动态的:Java 语言的设计目标之一是适应于动态变化的环境。Java 程序需要的类能够动态地被载入到运行环境,也可以通过网络来载入所需要的类。这也有利于软件的升级。另外,Java 中的类有一个运行时刻的表示,能进行运行时刻的类型检查。

62,615

社区成员

发帖
与我相关
我的任务
社区描述
Java 2 Standard Edition
社区管理员
  • Java SE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧