我们知道,在很多脚本语言中都有eval函数,它可以把字符串转换为表达式并执行.如在javaScript中:

var str = aid.value + ".style.top = 10;" 

 
把一个id为"aid"的控制的值取出来加合并成一个字符串,如果aid的值是"axman",则

     str = "axman.style.top = 10"  
现在我们要让控制axman移动到顶部为10的位置:

eval(str);  
这样这个字符串就变成了表达式或语句开始执行.这样的功能对于动态构造变量是有非常重要的
意义.

  那么在java中,如果实现这个功能呢?其实我们可以用动态编译来实现。

动态编译实现eval
  假设有一组方法实现不同的功能,现在要根据传进来的方法名调用相应的方法,假如没有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{";
+= "Object rt(){"
+= "MyClass mc = new MyClass();"
        s 
+= " return mc."+str+"();";
        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(nullnew 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文件

 完整的列子:

import java.io.File;
import java.io.FileWriter;
import java.io.PrintWriter;
import java.lang.reflect.Method;

import com.sun.tools.javac.Main;


public class testJavac{
        
public String getName(){
           
return "刘凯毅";
        }

        
public int getAvg(){
           
return 24;
        }

        
        
public Object eval(String str)throws Exception{
           
//生成java文件
        String s = "class Temp{";
            s 
+= "private testJavac tj = new testJavac();";
            s 
+= "public String rt(){";
            s 
+= " return  \"\"+tj."+str+"();"  ;
            s 
+= "}";
            s 
+="}";
            
           File f 
= new File(System.getProperty("user.dir")+"\\Temp.java");
           PrintWriter pw 
= new PrintWriter(new FileWriter(f));
           pw.println(s);
           pw.close();
           
//动态编译
           Main javac = new Main();
           String[] cpargs 
= new String[] {"-d", System.getProperty("user.dir") ,"Temp.java"};
           
int status = javac.compile(cpargs);
           
if(status!=0){
              System.out.println(
"没有成功编译源文件!");
              
return null;
           }

           
//调用Temp的rt方法返回结果:
           ClassLoader mc = this.getClass().getClassLoader();
           
           Class clasz 
= mc.loadClass("Temp");

           Method rt 
= clasz.getMethod("rt"new Class[]{});
           
return rt.invoke(clasz.newInstance(), new Object[] { });
           
//如果方法没有返回就直接调用
        }

        
        
    
public static void main(String[]args)throws Exception{
        testJavac jj 
= new testJavac();
        System.out.println( jj.eval(args[
0]) );
    }


}