繁体   English   中英

从另一个 Java 程序运行一个 Java 程序

[英]Running a java program from another java program

我正在开发一个简单的 java 程序。 它只是编译并执行另一个 java 程序。 我正在使用 Runtime.exec() 函数来编译和运行。 编译没有问题。 但是当它运行时,如果第二个程序需要一个输入来从键盘读取,我不能从主进程给它。 我使用了 getOutputStream() 函数。 但它无济于事。 我将提供我的代码。

public class sam {  
    public static void main(String[] args) throws Exception {  
        try { 
             Process p = Runtime.getRuntime().exec("javac sam2.java");
             Process p2 = Runtime.getRuntime().exec("java sam2");
             BufferedReader in = new BufferedReader(  
                                new InputStreamReader(p2.getInputStream()));

             OutputStream out = p.getOutputStream();
             String line = null; 
             line = in.readLine();
             System.out.println(line);
             input=input+"\n";
             out.write(input.getBytes());
             p.wait(10000);
             out.flush();
        }catch (IOException e) {  
             e.printStackTrace();  
        }  
    }  
}  

这是我的主程序(sam.java)。

以下是sam2.java的代码

public class sam2 {  
public static void main(String[] args) throws Exception {  

    BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    String str; 
    System.out.println("Enter the number..\n");
    str = br.readLine(); 
    System.out.println(Integer.parseInt(str));

    }  
}  

没有问题,如果我的第二个程序只有打印语句。 但是当我必须从另一个人那里读到一些东西时,问题就出现了。

这有点奇怪,但是您可以在不分叉的情况下运行第二个程序。 只需调用其中的 main 方法即可。 所以忘记运行时部分并执行以下操作:

sam2.main(new String[0]);

当然这种方式你必须在编译时编译sam2

每个进程都需要被允许运行和完成。 Process#waitFor ,您可以使用Process#waitFor 同样,您需要同时使用流程的任何输出。 waitFor将阻塞,因此您需要使用Thread来读取输入(如果需要,将输出写入进程)

根据 java/class 文件的位置,您可能还需要指定一个起始文件夹,进程的执行可以从该文件夹开始。

使用ProcessBuilder更轻松地完成大部分工作

import java.io.File;
import java.io.IOException;
import java.io.InputStream;

public class CompileAndRun {

    public static void main(String[] args) {
        new CompileAndRun();
    }

    public CompileAndRun() {
        try {
            int result = compile("compileandrun/HelloWorld.java");
            System.out.println("javac returned " + result);
            result = run("compileandrun.HelloWorld");
        } catch (IOException | InterruptedException ex) {
            ex.printStackTrace();
        }
    }

    public int run(String clazz) throws IOException, InterruptedException {        
        ProcessBuilder pb = new ProcessBuilder("java", clazz);
        pb.redirectError();
        pb.directory(new File("src"));
        Process p = pb.start();
        InputStreamConsumer consumer = new InputStreamConsumer(p.getInputStream());
        consumer.start();

        int result = p.waitFor();

        consumer.join();

        System.out.println(consumer.getOutput());

        return result;
    }

    public int compile(String file) throws IOException, InterruptedException {        
        ProcessBuilder pb = new ProcessBuilder("javac", file);
        pb.redirectError();
        pb.directory(new File("src"));
        Process p = pb.start();
        InputStreamConsumer consumer = new InputStreamConsumer(p.getInputStream());
        consumer.start();

        int result = p.waitFor();

        consumer.join();

        System.out.println(consumer.getOutput());

        return result;        
    }

    public class InputStreamConsumer extends Thread {

        private InputStream is;
        private IOException exp;
        private StringBuilder output;

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

        @Override
        public void run() {
            int in = -1;
            output = new StringBuilder(64);
            try {
                while ((in = is.read()) != -1) {
                    output.append((char) in);
                }
            } catch (IOException ex) {
                ex.printStackTrace();
                exp = ex;
            }
        }

        public StringBuilder getOutput() {
            return output;
        }

        public IOException getException() {
            return exp;
        }
    }
}

现在很明显,您应该检查进程的返回结果,并且可能会产生更好的与进程交互的机制,但这是基本思想......

你可以只调用第二个类的 main 方法。 main 方法就像任何其他静态方法一样。

这对我有用:

try {
    single.main(new String[0]);
} catch (Exception e) {
    JOptionPane.showMessageDialog(null, e);
}

只需调用主类文件。 例如,如果您的 java 类文件名为xyz.java ,则您可以在 java swing 应用程序中通过单击JButton调用并执行相同的文件,代码为

private void Btn_createdatabaseActionPerformed(java.awt.event.ActionEvent evt) {                                                   
       xyz.main(new String[0]);
}

而已...

暂无
暂无

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

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