没有等待结果如何执行sh文件?

时间:2013-08-28 15:22:55

标签: java linux runtime.exec

我有一个问题,我尝试从java代码运行一个sh文件来启动一个JBoss服务器,所以当我执行这个sh文件时,我想知道它何时启动并打印出我的控制台。这是我的代码:

    public static void runStart() {
        try {
            String command = "./Run.sh";
            String s = get_commandline_results(command);
            System.out.println(s);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("done");
    }

    public static String get_commandline_results(String cmd)
            throws IOException, InterruptedException, IllegalCommandException {

        String result = "";
        Process p = null;
        p = Runtime.getRuntime().exec(String.format("bash -c %s", cmd));
        final ProcessResultReader stderr = new ProcessResultReader(
                p.getErrorStream(), "STDERR");
        final ProcessResultReader stdout = new ProcessResultReader(
                p.getInputStream(), "STDOUT");
        System.out.println("starting...");
        stderr.start();
        stdout.start();
        final int exitValue = p.waitFor();// It was delayed here because sh not complete
        System.out.println("started!");// How to go to here?
        if (exitValue == 0) {
            result = stdout.toString();
        } else {
            result = stderr.toString();
        }
        return result;
    }   
}

class ProcessResultReader extends Thread {
    final InputStream is;
    final String type;
    final StringBuilder sb;

    ProcessResultReader(final InputStream is, String type) {
        this.is = is;
        this.type = type;
        this.sb = new StringBuilder();
    }

    public void run() {
        try {
            final InputStreamReader isr = new InputStreamReader(is);
            final BufferedReader br = new BufferedReader(isr);
            String line = null;
            while ((line = br.readLine()) != null) {
                this.sb.append(line).append("\n");
            }
        } catch (final IOException ioe) {
            System.err.println(ioe.getMessage());
            throw new RuntimeException(ioe);
        }
    }

    @Override
    public String toString() {
        return this.sb.toString();
    }
}

感谢您的帮助!

1 个答案:

答案 0 :(得分:2)

你在这里有相反的目标:你想继续而不等待脚本终止(即你希望JBoss运行),但同时,你想知道结果。没有建立时间机器,这是不可能的。

您可以做的是您可以在另一个脚本start.sh中启动脚本:

#!/bin/bash

nohup base -c ./Run.sh > /dev/null &

注意:您可以通过bash -c可执行文件来省略Run.sh

此脚本将作为后台进程启动Run.sh并立即退出。但是如果JBoss无法启动,你就不会收到错误,因为这会在以后发生。

解决这个难题的一个解决方案是一个脚本,它会在几秒钟内读取JBoss的日志文件,或直到它看到一行文本意味着“JBoss已成功启动”。

为此,使用启动脚本,然后在Java代码中,开始阅读JBoss的日志文件(您可能需要等待它首先显示)。

对于这种技巧,在尝试启动JBoss之前首先删除旧的日志文件总是好的。否则,Java程序可能会读取旧的日志文件,并在JBoss仍在尝试开始编写新日志时继续。