本文省略了异常
1.生成对象:
Process process = getObjectAsClassName("test.Process");
// 根据classname来生成对象 public static <U extends IProcess> U getObjectAsClassName(String className) { U proess = null; proess = (U) Class.forName(className).newInstance(); return proess; }
2、通过方法名执行方法:
Process process = getObjectAsClassName("test.Process");
callMehod("process", process);
// 根据method方法名执行方法 public static void callMehod(String methodName, Process p) { Method[] methods = Process.class.getMethods(); //Method[] methods = Class.forName("com.Process").getMethods(); for (Method method : methods) { if (method.getName().equals(methodName)) { // 找到方法,执行 method.invoke(p, new Object[] {}); } }
method.invoke():
第一个参数是对象,第二个参数是参数列表
3、基于接口的JDK动态代理:
IProcess processxy = getProxyProcess(process);
processxy.process();
//创建代理对象 @SuppressWarnings("unchecked") public static <U extends IProcess> U getProxyProcess(Object process) { ProcessProxy pp = new ProcessProxy(process); return (U) Proxy.newProxyInstance(Process.class.getClassLoader(), Process.class.getInterfaces(), pp); }
Proxy.newProxyInstance:
第一个参数是代理类的ClassLoader ,第二个参数是Interfaces,第三个是InvocationHandler对象
ProcessProxy.java:
public class ProcessProxy implements InvocationHandler { private Object targs; public ProcessProxy(Object targs) { this.targs = targs; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("被代理对象执行前"); Object ret = method.invoke(targs, args); System.out.println("被代理对象执行后"); return ret; } }
4、动态生成java文件并编译为class文件执行
// 生成动态类 //运行时动态生成文本的.java文件,调用java编译工具类编译成.class二进制文件,就像web服务器将jsp编译成servlet一样 @SuppressWarnings("unchecked") public static IProcess makeDynamicClassObject(IProcess p) { String _package = "package test;\r\n"; String classDefine = "public class DynamicClass implements IProcess {" + "\r\n" + "private IProcess p;\r\n" + "public DynamicClass(IProcess p) {\r\n" + "this.p = p;" + "}\r\n" + "public void process() {\r\n" + "System.out.println(\"动态生成的java类,开始执行\");" + "p.process();\r\n" + "}" + "}"; String javaSourceCode = _package + classDefine; System.out.println("System.getProperty('user.dir'):"+System.getProperty("user.dir")); //获取项目的根路径 String fileName = System.getProperty("user.dir") + "/src/test/DynamicClass.java"; // 设置一个路径 File javaFile = new File(fileName); try { // 创建java文件(写入文件) FileWriter outFile = new FileWriter(javaFile); outFile.write(javaSourceCode); outFile.flush(); outFile.close(); } catch (IOException e) { e.printStackTrace(); } // 生成class文件 JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); //此处需要使用JDK包,而非jre包,因为jre包中无编译器 StandardJavaFileManager fileMang = compiler.getStandardFileManager( null, null, Charset.forName("utf-8")); Iterable units = fileMang .getJavaFileObjects(fileName); CompilationTask t = compiler.getTask(null, fileMang, null, null, null, units); t.call(); fileMang.close(); javaFile.delete(); //执行class文件 URL url = new URL("file://" + System.getProperty("user.dir") + "/src/"); URLClassLoader uloader = new URLClassLoader(new URL[] { url }); Class c = uloader.loadClass("test.DynamicClass"); return (IProcess) c.getConstructor(IProcess.class).newInstance(p); return null; }
问题1:JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();出现空指针异常的问题
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();可以获得环境下边的编译器的引用,但为什么我们在debug到这一语句的时候,查看到此函数返回的是一个null值,即没有编译器。。。。
我们知道,jre是java runtime environment,我们在运行java程序的时候其实就是调用jre里面的java.exe或者javaw.exe,我们安装jdk1.6的时候,会在安装包java下发现jdk1.6和jre1.6两个文件夹,在jdk1.6下有 一个jre包,包里的内容和jre1.6文件夹下的内容是一样的。这是因为我们在开发程序的时候需要用到jre,所以jdk1.6.0_07下 有一个jre包,但只运行java程序的时候,只会用到jre1.6.0_07下的命令。
弄清楚这一点,问题就好说了,我们要获得编译器,可是jre1.6.0_07文件夹下根本没有编译器,所有我们应该使用jdk1.6.0_07文件 夹下的jre,在windows=》pereference=》java=>installl jre里面,添加jdk1.6中的jre,并勾选它。这样空指针异常就解决了。。。。
为什么我们用命令行执行相同的程序不会出现那个空指针异常呢,那是因为我们在寻找jre的时候是在优先考虑jdk下的jre的,但eclipse下是指定了jre的,所有我们在命令行下不会出错。。