Process.destroy()不会杀死Process的子级

时间:2016-06-16 11:09:59

标签: java linux bash ksh processbuilder

ProcessBuilder不支持bash重定向,因此如果我运行以下内容:

import java.lang.ProcessBuilder.Redirect;
import java.util.ArrayList;
import java.util.List;


public class Test {

public static void main(String args[]) {
 ProcessBuilder builder = new ProcessBuilder();
 String command = "ping 127.0.0.1 > console.log";
 builder.command("bash"  , "-c", command); 
 //builder.redirectOutput(Redirect.appendTo(new File ("console.log")));
 System.out.println("Builder: " + builder.command().toString());

try {
  Process p = builder.start();
      System.out.println("Sleeping for 20 seconds");
      Thread.sleep(20000);
      p.destroy(); 
} catch (Exception e) {
      System.out.println("An error happened here");
    }

}

}

p.destroy()只会杀死父bash -c进程,但不会杀死子进程。还有什么方法可以杀死孩子吗? ping将在destroy之后继续运行。

根据这篇文章Java - Process.destroy() source code for Linux,它最终会在本机c代码级别调用kill。

但即使我在Linux中这样做也不起作用:

[john@dub-001948-VM01:~/tmp ]$ bash -c 'ping 127.0.0.1 > console.log' &
[1] 30914
[john@dub-001948-VM01:~/tmp ]$ ps
 PID TTY          TIME CMD
30536 pts/1    00:00:00 bash
30914 pts/1    00:00:00 bash
30915 pts/1    00:00:00 ping
30916 pts/1    00:00:00 ps
[john@dub-001948-VM01:~/tmp ]$ kill -9 30914
[john@dub-001948-VM01:~/tmp ]$ ps -ef | grep ping
john     30915     1  0 15:19 pts/1    00:00:00 ping 127.0.0.1
john     30919 30536  0 15:19 pts/1    00:00:00 grep ping
[1]+  Killed                  bash -c 'ping 127.0.0.1 > console.log'
[john@dub-001948-VM01:~/tmp ]$ 

ping仍在运行..

1 个答案:

答案 0 :(得分:1)

看起来bash执行fork和exec来启动子进程,因此不会杀死子进程。但是,如果您使用ksh而不是bash,那么您将获得一个没有exec的分支:

ksh -c ping 127.0.0.1 > console.log

使用kshprocess.destroy()会杀死ksh和子进程

我不确定,-cksh的手册页中的bash选项看起来非常相似。