62,615
社区成员
发帖
与我相关
我的任务
分享
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加载进来,这样,在做强制类型转换的时候,就是同一个接口类了.
*/
/*
* @(#)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;
}
}
}
/*
* @(#)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!");
}
}
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;
}
}