一篇参考博客:http://www.cnblogs.com/fangwenyu/archive/2011/10/12/2209051.htmlhtml
在Python中有一个exec()函数,一样在JavaScript中有一个eval()函数,这两个函数有一个类似的特色,那就是能够在里面传入一段Python代码或者JavaScript代码,发现居然能够运行该代码。java
可是遗憾的是,Java中并不存在这样的函数,因而突发奇想,咱们可不能够在Java中实现一个相似的函数,用来执行Java代码呢?linux
咱们知道,Python和JavaScript属于脚本语言,也就是非编译型语言,它们并不存在先编译后执行的过程的。而Java、C++这种编译型的语言,通常都是先编译,后执行。对于Java来讲,编译生成.class文件,而后JVM运行.class文件。而咱们若是想要将Java代码传入方法中,而后运行,那么就不能采用传统的编译+运行了,采用Java提供的动态编译和动态加载的机制。数据库
下面我经过一个简单的例子加以说明,写一个eval()方法,能够执行System.out.println("Hello, " + str);这行代码:windows
Eval.java:数组
package com.darrenchan; import java.lang.reflect.Method; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import javax.tools.JavaCompiler; import javax.tools.JavaFileObject; import javax.tools.SimpleJavaFileObject; import javax.tools.StandardJavaFileManager; import javax.tools.ToolProvider; public class Eval { /* * 从java6版本开始,已经支持动态编译了,你能够在运行期直接编译.java文件,执行.class文件,而且可以得到相关的输入输出, * 甚至还能监听相关的事件。 * java的动态编译提供了多种渠道,好比,能够动态编译一个字符串,也能够是文本文件,也能够是编译过的字节码文件(.class文件), * 甚至能够是存放在数据库中的明文代码或字节码,只要是符合java规范的就均可以在运行期动态加载,其实现方式就是实现JavaFileObject * 接口,重写getCharContent、openInputStream、openOutputStream,或者实现JDK * 已经提供的两个SimpleJavaFileObject、ForwardingJavaFileObject。下面我演示一下,如何动态编译一个字符串。 */ /** * Java动态编译演示 */ public static void main(String[] args) throws Exception { // Java源代码 String sourceStr = "public class Hello{public String sayHello(String name){return \"Hello, \"+name;}}"; // 类及文件名 String clsName = "Hello"; // 方法名 String methodName = "sayHello"; /** * 当前编译器:注意,若是是用的jdk1.6的版本(建议使用jdk1.7,1.7是没有任何问题的),ToolProvider. * getSystemJavaCompiler()拿到的对象将会为null, * 缘由是须要加载的Tools.jar不在jdk安装目录的jre目录下,须要手动将lib目录下的该jar包拷贝到jre下去,详情请参考: * http://www.cnblogs.com/fangwenyu/archive/2011/10/12/2209051.html */ JavaCompiler cmp = ToolProvider.getSystemJavaCompiler(); // Java标准文件管理器 StandardJavaFileManager fm = cmp.getStandardFileManager(null, null, null); // Java文件对象 JavaFileObject jfo = new StringJavaObject(clsName, sourceStr); // 编译参数,相似于javac <options> 中的options List<String> optionsList = new ArrayList<String>(); // 编译文件的存放地方,注意:此处是为Eclipse工具特设的 optionsList.addAll(Arrays.asList(new String[] { "-d", "./bin" })); // 要编译的单元 List<JavaFileObject> jfos = Arrays.asList(new JavaFileObject[] { jfo }); // 设置编译环境 JavaCompiler.CompilationTask task = cmp.getTask(null, fm, null, optionsList, null, jfos); // 编译成功 if (task.call()) { // 生成对象 Object obj = Class.forName(clsName).newInstance(); Class<? extends Object> cls = obj.getClass(); // 调用sayHello方法 Method m = cls.getMethod(methodName, String.class); // 第一个参数是执行该方法的主调,后面若干个参数是执行该方法时传入该方法的实参 String str = (String) m.invoke(obj, "陈驰"); System.out.println(str); } } }
StringJavaObject.java:ide
package com.darrenchan; import java.io.IOException; import java.net.URI; import javax.tools.SimpleJavaFileObject; public class StringJavaObject extends SimpleJavaFileObject { /** * 源代码 */ private String content = ""; /** * 遵循Java规范的类名及文件 */ public StringJavaObject(String javaFileName, String content){ super(_createStringJavaObjectUri(javaFileName), Kind.SOURCE); this.content = content; } /** * 产生一个URL资源路径 */ private static URI _createStringJavaObjectUri(String javaFileName) { //注意此处未设置包名 return URI.create("String:///" + javaFileName + Kind.SOURCE.extension); } /** * 文本文件代码 */ @Override public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException { return content; } }
通过测试,最终的运行结果符合预期,以下所示:函数