在java中听stdin的后台进程

时间:2012-07-25 06:11:38

标签: java shell stdin background-process

我必须创建一个java程序,当用户输入0时它应该退出。用java代码编写没问题。

int cmd = read();
System.out.println("got command : " + cmd);
if (cmd == 48) { // ASCII code for 0
System.exit(0);

我想在linux中使用start-stop脚本运行此过程。我也可以使用&nohup

来做到这一点
case "$1" in    
  'start')
    if [ -f myfifo ]; then
      rm myfifo
    fi
    mkfifo myfifo
    cat > myfifo &
    echo $! > myfifo-cat-pid
    java -jar myjar.jar >/dev/null 2>&1 0<myfifo &
    echo "Started process: "$!
    ;;

  'stop')
    echo 0 > myfifo
    echo "Stopped process: "
    rm myfifo
    ;;
esac

我的问题是,只要我运行此过程,它就会读取输入-1。虽然我想从stdinput中读取somthing明确地回显它,即调用stop。我只需要一个程序由shell脚本显式关闭。我努力了。帮助我。

编辑:

我只想在用户按q或任何此类事件时关闭程序,并且不希望代码中出现无限循环。如果可能的话,任何其他方法如听其他事件也会有所帮助。我的代码不应该轮询事件。我已经尝试Jnotify监视目录并在该目录中创建或删除文件时引发事件。但幸运的是,他们没有对SunOS的支持。 :(

4 个答案:

答案 0 :(得分:3)

嗯,实际上fifo文件看起来并不像你想象的那样。

read()命令未在管道上阻塞。您应该循环直到获得数据:

    try {
        bis = new BufferedInputStream(System.in);
        while (true) {
            byte[] buffer = new byte[4096];
            if (bis.available() > 0) {
                bis.read(buffer, 0, bis.available());
                // do some clever thing
            } else {
                try {
                    Thread.sleep(50);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    } catch (IOException e) {
        // it failed...
    }

这将尝试读取,如果没有,则等待50 ms。

微米。

答案 1 :(得分:1)

由于管道中还没有数据,read()会看到流的结尾,因此会返回-1。在从流中读取之前调用available()以确保其中包含信息。

答案 2 :(得分:0)

如果您在shell脚本的顶部使用set -m(因此强制执行作业控制),您应该会看到一致的行为。

答案 3 :(得分:0)

当您在脚本中显示命令时,该命令的标准输入将从/dev/null隐式重定向。

# test by executing a script with ...
- cat > myfifo & lsof -p $!
+ cat 0<&0 > myfifo & lsof -p $!