java 类 重新加载_Java动态编译和类的重新加载 | 学步园

注意:

1.动态编译的class文件不要放到jre的ClassPath中,在jre在ClassPath中找到的类只会加载一次。

2.使用反射获取类的时,如果需要获取最新生成的,需要重新实例化一个类加载器,因为旧的类加载器已经加载过这个类,再次加载只会加载上次那个。而重新实例化的类加载器没有加载过这个类,所以会重新去定义、链接和加载。

eeb8516aff38c4a7887cf4c914789f4f.png

Main.java

public class Main {

public static void main(String[] args) {

String fullClassName = "MyObj";

String code = "public class MyObj implements MyInterface{public void sayHello(){System.out.println(666);}}";

String code_2 = "public class MyObj implements MyInterface{public void sayHello(){System.out.println(777);}}";

String code_3 = "public class MyObj implements MyInterface{public void sayHello(){System.out.println(888);}}";

load(code,fullClassName);

load(code_2,fullClassName);

load(code_3,fullClassName);

}

private static void load(String code, String fullClassName) {

new MyClassCompiler(fullClassName, code).compile();

try {

MyInterface myObj = (MyInterface) new MyClassLoader().loadClass(

fullClassName).newInstance();

myObj.sayHello();

} catch (InstantiationException e) {

// TODO Auto-generated catch block

e.printStackTrace();

} catch (IllegalAccessException e) {

// TODO Auto-generated catch block

e.printStackTrace();

} catch (ClassNotFoundException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

MyClassCompiler.java

import java.io.File;

import java.io.IOException;

import java.net.URI;

import java.net.URISyntaxException;

import java.util.Arrays;

import javax.tools.JavaCompiler;

import javax.tools.JavaCompiler.CompilationTask;

import javax.tools.JavaFileObject;

import javax.tools.JavaFileObject.Kind;

import javax.tools.SimpleJavaFileObject;

import javax.tools.ToolProvider;

public class MyClassCompiler {

private String simpleClassName;

private String code;

private String classPath = System.getProperty("user.dir") + File.separator + "myFolder";

public MyClassCompiler(String simpleClassName, String code) {

this.simpleClassName = simpleClassName;

this.code = code;

}

public boolean compile() {

try {

JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();

JavaFileObject javaFile = new SimpleJavaFileObject(new URI(

simpleClassName + ".java"), Kind.SOURCE) {

@Override

public CharSequence getCharContent(boolean arg)

throws IOException {

return code;

}

};

CompilationTask task = compiler.getTask(null, null, null,

Arrays.asList("-d", classPath), null,

Arrays.asList(javaFile));

return task.call();

} catch (URISyntaxException e) {

e.printStackTrace();

return false;

}

}

public String getSimpleClassName() {

return simpleClassName;

}

public void setSimpleClassName(String simpleClassName) {

this.simpleClassName = simpleClassName;

}

public String getCode() {

return code;

}

public void setCode(String code) {

this.code = code;

}

public String getClassPath() {

return classPath;

}

public void setClassPath(String classPath) {

this.classPath = classPath;

}

}

MyClassLoader.java

import java.io.File;

import java.io.FileInputStream;

import java.io.IOException;

import java.io.InputStream;

public class MyClassLoader extends ClassLoader {

private boolean alwaysDefineClass = true;

private String classPath = System.getProperty("user.dir") + File.separator + "myFolder";

@Override

protected Class> findClass(String fullClassName)

throws ClassNotFoundException {

Class> clazz = null;

// clazz = findLoadedClass(fullClassName);

// if(alwaysDefineClass || clazz == null){

byte[] raw = readClassBytes(fullClassName);

clazz = defineClass(fullClassName, raw, 0, raw.length);

resolveClass(clazz);

// }

return clazz;

}

private byte[] readClassBytes(String fullClassName) {

byte[] raw = null;

InputStream stream = null;

File file = new File(classPath + File.separator

+ fullClassName.replaceAll("\\.", "/") + ".class");

try {

stream = new FileInputStream(file);

raw = new byte[(int) file.length()];

stream.read(raw);

} catch (Exception e) {

} finally {

try {

stream.close();

} catch (IOException e) {

}

}

return raw;

}

public boolean isAlwaysDefineClass() {

return alwaysDefineClass;

}

public void setAlwaysDefineClass(boolean alwaysDefineClass) {

this.alwaysDefineClass = alwaysDefineClass;

}

public String getClassPath() {

return classPath;

}

public void setClassPath(String classPath) {

this.classPath = classPath;

}

}

interface.java

public interface MyInterface {

public void sayHello();

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 中,动态加载是指在程序运行时动态加载,而不是在编译加载。这种方式可以使程序更加灵活,可以根据不同的需求加载不同的Java 中的枚举(Enum)也可以通过动态加载的方式来实现。枚举是一种特殊的,可以包含常量、方法和字段。在 Java 中,枚举常量是在编译时确定的,但是可以通过动态加载动态加载枚举。 下面是一个动态加载枚举的示例代码: ```java import java.lang.reflect.Method; import java.net.URL; import java.net.URLClassLoader; public class DynamicEnumLoader { public static void main(String[] args) throws Exception { // 枚举的路径 URL url = new URL("file:/path/to/enum/classes"); // 创建加载器 URLClassLoader classLoader = new URLClassLoader(new URL[]{url}); // 加载枚举 Class<?> enumClass = classLoader.loadClass("com.example.MyEnum"); // 获取枚举常量 Object[] enumConstants = enumClass.getEnumConstants(); // 调用枚举方法 Method method = enumClass.getMethod("getValue"); for (Object obj : enumConstants) { String value = (String) method.invoke(obj); System.out.println(value); } } } ``` 在上面的示例中,我们创建了一个 URLClassLoader 对象,将枚举的路径传入其中。然后使用该加载加载了 MyEnum 枚举。最后,通过反射的方式调用了 getValue 方法,获取枚举常量的值。 需要注意的是,动态加载的枚举必须要和程序的在同一个加载器中,否则会出现 ClassCastException 异常。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值