2014-11-06 113 views
1

我已经使用JavaCompiler来编译我在运行中生成的类。编译任务成功。然后我尝试使用Class.forName("MyClass")加载编译的类;并且它以ClassNotFoundException失败。这样做有没有已知的问题?未找到JavaCompiler编译类

package com.amir.method; 

import java.io.*; 
import java.lang.reflect.Method; 
import java.net.URI; 
import java.net.URL; 
import java.net.URLClassLoader; 
import java.util.ArrayList; 
import java.util.Arrays; 
import java.util.List; 
import java.util.Set; 

import javax.tools.*; 

public class MethodGenerator { 

    private final static MethodGenerator INSTANCE = new MethodGenerator(); 

    private MethodGenerator() { 
    } 

    public static MethodGenerator get() { 
     return INSTANCE; 
    } 


    public static void main(String[] args) { 

     final Class<?> clz = MethodGenerator.get().compile("Foo", "doIt"); //<- args ignored for now 

    } 

    public Class compile(final String className, 
         final String methodName) { 
     try { 
      return doCompile(className, methodName); 
     } catch(final Exception e) { 
      throw new RuntimeException(this.toString(), e); 
     } 

    } 

    private Class doCompile(final String className, 
          final String methodName) throws IOException, ClassNotFoundException { 

     JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); 
     DiagnosticCollector<JavaFileObject> diagnostics = new DiagnosticCollector<JavaFileObject>(); 

     StringWriter writer = new StringWriter(); 
     PrintWriter out = new PrintWriter(writer); 
     out.println("public class HelloWorld {"); 
     out.println(" public static void main(String args[]) {"); 
     out.println(" System.out.println(\"This is in another java file\");"); 
     out.println(" }"); 
     out.println("}"); 
     out.close(); 

     final JavaFileObject file = new JavaSourceFromString("HelloWorld", writer.toString()); 
     final Iterable<? extends JavaFileObject> compilationUnits = Arrays.asList(file); 
     final String classPath = System.getProperty("java.class.path") + File.pathSeparator; 
     final String bootClassPath = System.getProperty("sun.boot.class.path") + File.pathSeparator; 
     final List<String> options = new ArrayList<String>(); 
     options.add("-classpath"); 
     final StringBuilder builder = new StringBuilder(); 
     builder.append(classPath); 
     builder.append(bootClassPath); 
     final URLClassLoader urlClassLoader = (URLClassLoader) ClassLoaderResolver.getClassLoader(); 
     for (final URL url : urlClassLoader.getURLs()) { 
      builder.append(url.getFile()).append(File.pathSeparator); 
     } 
     final int lastIndexOfColon = builder.lastIndexOf(File.pathSeparator); 
     builder.replace(lastIndexOfColon, lastIndexOfColon + 1, ""); 
     options.add(builder.toString()); 

     final JavaCompiler.CompilationTask task = compiler.getTask(null, null, diagnostics, options, null, compilationUnits); 

     boolean success = task.call(); 

     if(success) { 
      final Class<?> cls = Class.forName("HelloWorld", true, urlClassLoader); 
      return cls; 
     } 

     return null; 

    } 

    class JavaSourceFromString extends SimpleJavaFileObject { 
     final String code; 

     JavaSourceFromString(String name, String code) { 
      super(URI.create("string:///" + name.replace('.', '/') + Kind.SOURCE.extension),Kind.SOURCE); 
      this.code = code; 
     } 

     @Override 
     public CharSequence getCharContent(boolean ignoreEncodingErrors) { 
      return code; 
     } 
    } 

} 
+0

您是否检查过您的清单,如果导出/导入该类?你如何建立你的项目。尝试重新生成清单文件。 – StackFlowed 2014-11-06 17:56:56

+0

清单?我正在编译一个课程。我不确定我明白清单文件会做什么? – 2014-11-06 17:58:37

+0

编译该类不会自动将其添加到类加载器。你需要自己做。 – Paul 2014-11-06 18:02:42

回答

0

尝试通过编译之后加载类,Thread.currentThread().getContextClassLoader().loadClass("com.ur.class");

发现这个...检查是否这可以帮助..

// Create a new custom class loader, pointing to the directory that contains the compiled 
// classes, this should point to the top of the package structure! 
URLClassLoader classLoader = new URLClassLoader(new URL[]{new File("./").toURI().toURL()}); 
// Load the class from the classloader by name.... 
Class<?> loadedClass = classLoader.loadClass("com.ur.class"); 
// Create a new instance... 
Object obj = loadedClass.newInstance(); 
// Santity check 
if (obj instanceof com.ur.class) { 
    // code here ... 
} 

欲了解更多信息,请参阅本:How do you dynamically compile and load external java classes?

+0

字符串参数究竟是什么?在评论部分,有人建议我应该使用“com.amir.method.HelloWorld”,但这似乎不起作用... – 2014-11-06 18:09:49

+0

@AmirAfghani该班有什么包? – 2014-11-06 18:15:06

+0

@VinceEmigh - 从生成的代码中可以看出,它不在任何特定的包中。我的来源没有包装声明。 – 2014-11-06 18:17:20

相关问题