简体   繁体   English

如何获取java程序的输出?

[英]how to get the output of a java program?

I need to write a code that could help me to get the output of a program. 我需要编写一个代码来帮助我获取程序的输出。

I have in that Java file MainClass.java that I am compiling, a simple code that prints "OK" . 我在那个正在编译的Java文件MainClass.java中,有一个打印"OK"的简单代码。

How could I return this output and even if there is error, I need to take it as well. 我怎么能返回这个输出,即使有错误,我也需要把它拿走。

This is my code that compile and create a .class file. 这是我编译和创建.class文件的代码。

File sourceFile = new File("C:\\Projet_Interne\\webProject\\NewFolder\\MainClass.java");  
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
int compilationResult = compiler.run(null, null, null,sourceFile.getPath());
int result = compiler.run(System.in,System.out,System.err,sourceFile.getPath());
     System.out.println("Compile result code = " + result);
      if(compilationResult == 0){
            System.out.println("Compilation is successful");
      }else{
             System.out.println("Compilation Failed");
      }
 StandardJavaFileManager stdFileManager = compiler.getStandardFileManager(null, Locale.getDefault(), null);
 stdFileManager.setLocation(StandardLocation.CLASS_OUTPUT, Arrays.asList(new File("C:\\Projet_Interne\\webProject\\NewFolder")));

The run method can use streams to handle input and output: run方法可以使用流来处理输入和输出:

int run(InputStream in,
      OutputStream out,
      OutputStream err,
      String... arguments)

Use an OutputStream for out and read from it when run is finished. 使用OutputStream out并在run完成后从中读取。

You could use the DiagnosticCollector , which will allow you to collect diagnostic information about the compilation process, you can find more details from the JavaDocs 您可以使用DiagnosticCollector ,它允许您收集有关编译过程的诊断信息,您可以从JavaDocs中找到更多详细信息

For example... 例如...

File helloWorldJava = new File(...);

DiagnosticCollector<JavaFileObject> diagnostics = new DiagnosticCollector<JavaFileObject>();
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
StandardJavaFileManager fileManager = compiler.getStandardFileManager(diagnostics, null, null);

// This sets up the class path that the compiler will use.
// I've added the .jar file that contains the DoStuff interface within in it...
List<String> optionList = new ArrayList<String>();
optionList.add("-classpath");
optionList.add(System.getProperty("java.class.path"));

Iterable<? extends JavaFileObject> compilationUnit
        = fileManager.getJavaFileObjectsFromFiles(Arrays.asList(helloWorldJava));
JavaCompiler.CompilationTask task = compiler.getTask(
    null, 
    fileManager, 
    diagnostics, 
    optionList, 
    null, 
    compilationUnit);
if (task.call()) {
    System.out.println("Yipe");
} else {
    // Opps compile failed...
    for (Diagnostic<? extends JavaFileObject> diagnostic : diagnostics.getDiagnostics()) {
        System.out.format("Error on line %d in %s%n",
                diagnostic.getLineNumber(),
                diagnostic.getSource().toUri());
    }
}
fileManager.close();

Once compiled, you have two choices, you can either use a custom class loader to load the class and execute it, which will use your current stdout... 编译完成后,您有两个选择,您可以使用自定义类加载器来加载类并执行它,这将使用您当前的标准输出...

// 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("testcompile.HelloWorld");
// Create a new instance...
Object obj = loadedClass.newInstance();
// Santity check
if (obj instanceof DoStuff) {
    // Cast to the DoStuff interface
    DoStuff stuffToDo = (DoStuff)obj;
    // Run it baby
    stuffToDo.doStuff();
}

Or use a ProcessBuilder to execute another Java process... 或使用ProcessBuilder执行另一个Java进程......

ProcessBuilder pb = new ProcessBuilder("java", "HelloWorld");
pb.directory(new File("src"));
pb.redirectError();
Process p = pb.start();

InputStreamConsumer.consume(p.getInputStream());

p.waitFor();

For reference, the InputStreamConsumer .... 作为参考, InputStreamConsumer ....

public static class InputStreamConsumer implements Runnable {

    private InputStream is;

    public InputStreamConsumer(InputStream is) {
        this.is = is;
    }

    public InputStream getInputStream() {
        return is;
    }

    public static void consume(InputStream is) {
        InputStreamConsumer consumer = new InputStreamConsumer(is);
        Thread t = new Thread(consumer);
        t.start();
    }

    @Override
    public void run() {
        InputStream is = getInputStream();
        int in = -1;
        try {
            while ((in = is.read()) != -1) {
                System.out.print((char)in);
            }
        } catch (IOException exp) {
            exp.printStackTrace();
        }
    }

}

This is outlined in more detail in: 这在以下内容中有更详细的概述:

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM