在java中利用动态编译实现eval
2008-12-04 11:58
495 查看
我们知道,在很多脚本语言中都有eval涵数,它可以把字符串转换为表态式并执行.如在javaScript中
var str = aid.value + ".style.top = 10;"
把一个id为"aid"的控制的值取出来加合并成一个字符串,如果aid的值是"axman",则
str = "axman.style.top = 10"
现在我们要让控制axman移动到顶部为10的位置:
eval(str);
这样这个字符串就变成了表达式或语句开始执行.这样的功能对于动态构造变量是有非常重要的
意义.
那么在java中,如果实现这个功能呢?其实我们可以用动态编译来实现:
假设有一组方法实现不同的功能,现在要根据传进来的方法名调用相应的方法,假如没有eval功能,我们
只能这么做:
MyClass mc = new MyClass();
if(str.equals("m1"))
mc.m1();
else if(str.equals("m1"))
mc.m2();
else if(str.equals("m3"))
mc.m3();
else if(.........)
.........();
如果有一百种情况呢?
如果我们用eval方法就可以直接这样:
String str = ...........;
eval("mc"+str+"();");
是不是非常方便?关键是如何实现eval()?
我们把要转换的字符串构造一个完整的类:如果方法是有返回值的.则:
public Object eval(String str){
//生成java文件
String s = "class Temp{";
s += "Object rt(){"
s += "MyClass mc = new MyClass();"
s += " return mc."+str+"();";
s += "}"
s +="}";
File f = new File("Temp.java");
PrintWriter pw = new PrintWriter(new FileWriter(f));
pw.println(s);
pw.close();
//动态编译
com.sun.tools.javac.Main javac = new com.sun.tools.javac.Main();
String[] cpargs = new String[] {"-d", "所在目录","Temp.java"};
int status = javac.compile(cpargs);
if(status!=0){
System.out.println("没有成功编译源文件!");
return null;
}
//调用Temp的rt方法返回结果:
MyClassLoader mc = new MyClassLoader();
Class clasz = mc.loadClass("Test.class",true);
Method rt = clasz.getMethod("rt", new Class[]{ String[].class });
return rt.invoke(null, new Object[] { new String[0] });
//如果方法没有返回就直接调用
}
我们可以先写好多个重载的eval,有返回值和没有返回值的.以及可以传递参数的.
这样我们就可以用字符串转换为java的语句来执行.
本文只是一个例子,说明了一个动态编译的思想,更好的实现请各位朋友自己来完成.
关于动态编译的参数,补充说明一下:
String[] cpargs = new String[] {"-d", "所在目录","Temp.java"};
-d指明的目录应该是当前目录,因为生成的java文件是以当前目录为"/"然后在此目录下建立相应的包的."当前目录"应该用new File(".").getAbsoultPath()来确定.
java文件如果有package,在生成的时候应该建立相应的子目录.而
这个参数应该是 java源文件的file对象的getAbsoultPath(),
如当前应用程序是在d:/debug目录运行,动态生成的java文件有个package 为temp;
则 String[] cpargs = new String[] {"-d", "d://debud","d://debug//temp//Temp.java"};这样生成的class文件应该和java源文件在同一目录d:/debug//temp/下.
重载loadClass方法时应该注意能正确读取到class文件
var str = aid.value + ".style.top = 10;"
把一个id为"aid"的控制的值取出来加合并成一个字符串,如果aid的值是"axman",则
str = "axman.style.top = 10"
现在我们要让控制axman移动到顶部为10的位置:
eval(str);
这样这个字符串就变成了表达式或语句开始执行.这样的功能对于动态构造变量是有非常重要的
意义.
那么在java中,如果实现这个功能呢?其实我们可以用动态编译来实现:
假设有一组方法实现不同的功能,现在要根据传进来的方法名调用相应的方法,假如没有eval功能,我们
只能这么做:
MyClass mc = new MyClass();
if(str.equals("m1"))
mc.m1();
else if(str.equals("m1"))
mc.m2();
else if(str.equals("m3"))
mc.m3();
else if(.........)
.........();
如果有一百种情况呢?
如果我们用eval方法就可以直接这样:
String str = ...........;
eval("mc"+str+"();");
是不是非常方便?关键是如何实现eval()?
我们把要转换的字符串构造一个完整的类:如果方法是有返回值的.则:
public Object eval(String str){
//生成java文件
String s = "class Temp{";
s += "Object rt(){"
s += "MyClass mc = new MyClass();"
s += " return mc."+str+"();";
s += "}"
s +="}";
File f = new File("Temp.java");
PrintWriter pw = new PrintWriter(new FileWriter(f));
pw.println(s);
pw.close();
//动态编译
com.sun.tools.javac.Main javac = new com.sun.tools.javac.Main();
String[] cpargs = new String[] {"-d", "所在目录","Temp.java"};
int status = javac.compile(cpargs);
if(status!=0){
System.out.println("没有成功编译源文件!");
return null;
}
//调用Temp的rt方法返回结果:
MyClassLoader mc = new MyClassLoader();
Class clasz = mc.loadClass("Test.class",true);
Method rt = clasz.getMethod("rt", new Class[]{ String[].class });
return rt.invoke(null, new Object[] { new String[0] });
//如果方法没有返回就直接调用
}
我们可以先写好多个重载的eval,有返回值和没有返回值的.以及可以传递参数的.
这样我们就可以用字符串转换为java的语句来执行.
本文只是一个例子,说明了一个动态编译的思想,更好的实现请各位朋友自己来完成.
关于动态编译的参数,补充说明一下:
String[] cpargs = new String[] {"-d", "所在目录","Temp.java"};
-d指明的目录应该是当前目录,因为生成的java文件是以当前目录为"/"然后在此目录下建立相应的包的."当前目录"应该用new File(".").getAbsoultPath()来确定.
java文件如果有package,在生成的时候应该建立相应的子目录.而
这个参数应该是 java源文件的file对象的getAbsoultPath(),
如当前应用程序是在d:/debug目录运行,动态生成的java文件有个package 为temp;
则 String[] cpargs = new String[] {"-d", "d://debud","d://debug//temp//Temp.java"};这样生成的class文件应该和java源文件在同一目录d:/debug//temp/下.
重载loadClass方法时应该注意能正确读取到class文件
相关文章推荐
- Java中利用Interpreter动态编译实现eval
- 在java中利用动态编译实现eval
- 在java中利用动态编译实现eval
- 利用C#动态编译功能实现像Javascript中的Eval的功能来将一段字符串进行数学运算
- Java实现eval()方法,完成动态编译
- Java利用动态代理模拟实现Spring的AOP(面向切面编程)功能的小实践
- jfinal-coffeescript使用java实现coffeescript的动态编译
- 在Java中利用动态代理实现数据库连接与事务的自动管理
- 利用jquery,java实现表格动态分页
- 利用JEXL实现动态表达式编译
- Java 程序设计 利用线程实现动态显示系统时间
- 利用php中的eval来模拟java中的动态代理
- Java实现动态编译的两种方式
- java动态编译实现
- JAVAWEB开发之Servlet3.0新特性的使用以及注解的详细使用和自定义注解的方法、动态代理的使用、利用动态代理实现细粒度的权限控制以及类加载和泛型反射
- jfinal-lesscss使用java实现lesscss的动态编译
- JAVA1.6实现动态编译加载运行
- 在Java中利用动态代理实现数据库连接与事务的自动管理
- Java 程序设计 利用线程实现动态显示系统时间
- 如何在SpringMVC框架中利用Java反射机制和Javassist实现Java对象、属性、注解的动态创建生成