如果不在类文件中,则存储已编译的代码

时间:2014-02-06 12:55:51

标签: java compiler-construction jvm .class-file java-compiler-api

最近,我在阅读 THE STRUCTURE OF THE JAVA VIRTUAL MACHINE

时遇到了一个问题

第5页第5页2,

  

Java虚拟机要执行的编译代码是   使用独立于硬件和操作系统的二进制文件表示   格式,通常(但不一定)存储在文件中,称为   类文件格式。

支架但不一定是原因。

问题是,

在哪种情况下编译的代码不会存储在类文件中?如果它不会存储在类文件中,那么在哪里以及如何?

修改:请注意,问题与ClassLoader无关。

4 个答案:

答案 0 :(得分:3)

在java中,类加载器获取编译的二进制文件取决于它的实现。 可以编写一个类加载器,从数据库,网络,内存或任何其他可以想象的位置加载它的类。

默认的java URLClassLoader使用目录或jar中的文件,因此“通常”来自哪里,“不一定”只是提示可能有其他实现。

答案 1 :(得分:3)

你似乎对如何在内存中生成java类文件更感兴趣,所以这里是:

public class CompileSourceInMemory {
    public static void main(String args[]) throws IOException {
        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();
        JavaFileObject file = new JavaSourceFromString("HelloWorld", writer.toString());

        Iterable<? extends JavaFileObject> compilationUnits = Arrays.asList(file);
        CompilationTask task = compiler.getTask(null, null, diagnostics, null, null, compilationUnits);

        boolean success = task.call();

        if (success) {
            try {
                Class.forName("HelloWorld").getDeclaredMethod("main", new Class[] { String[].class }).invoke(null, new Object[] { null });
            } catch (ClassNotFoundException e) {
                System.err.println("Class not found: " + e);
            } catch (NoSuchMethodException e) {
                System.err.println("No such method: " + e);
            } catch (IllegalAccessException e) {
                System.err.println("Illegal access: " + e);
            } catch (InvocationTargetException e) {
                System.err.println("Invocation target: " + e);
            }
        }
    }
}

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;
    }
}

来源: In memory compilation

这会在内存中生成一个类,而不包含java源代码或编译类的任何外部表示。

答案 2 :(得分:2)

将编译后的代码存储在.class文件中只是获取JVM可执行代码的标准且更方便的方法,但是,可以从许多其他源获取相同的编译代码,它可以作为文本或二进制文件获取来自数据库或来自网络连接的流或来自运行时的内存,check this examplethis one

答案 3 :(得分:0)

是的,这个问题是关于Classloader的!

Classloader是JVM带来的(唯一)方式。这些类可以是文件,JAR存档,网络中的某个位置,数据库blob,共享内存,纸带(如果您的计算机上有纸质磁带阅读器),或者甚至可以即时生成。

所以,这是你问题的答案。

标准编译器用于编写类文件,但正如Peter所评论的那样,并不一定如此。而且,最重要的是,仅仅因为javac命令写入类文件并不意味着我们必须使用这些文件。通常将它们打包在JAR中,然后使用它。