Thread-launched running processes won't destroy (Java)

前端 未结 8 753
清歌不尽
清歌不尽 2021-02-01 08:41

Starting multiple threads and having each exec() then destroy() a running java process result in some of the process not being destroyed and still running after program exit. He

相关标签:
8条回答
  • 2021-02-01 09:19

    Use a p.waitFor(); before p.destroy(); ,

    this will ensure the completion of the previous process. I think you p.destroy command gets invoked sooner than the exec() command performs the action. Therefore it becomes useless.

    0 讨论(0)
  • 2021-02-01 09:21

    This is simply because before the threads execute the destroy call, your main program terminates and all the associated threads leaving the started processes running. To verify this, simply add a System.out call after the destroy and you will find it is not executed. To overcome this add a Thread.sleep at the end of your main method and you will not have the orphaned processes. The below does not leave any process running.

    public class ProcessTest {
    
    public static final void main (String[] args) throws Exception {
    
        for(int i = 0; i < 100; i++) {
            new Thread(new Runnable()
                {
                    public void run() {
                        try {
                            Process p = Runtime.getRuntime().exec(new String[]{"java", "InfiniteLoop"});
                            Thread.sleep(1);
                            p.destroy();
                            System.out.println("Destroyed");
                        }catch(IOException e) {
                            System.err.println("exception: " + e.getMessage());
                        } catch(InterruptedException e){
                            System.err.println("exception: " + e.getMessage());
                        }
                    }
                }).start();
        }
    
    
        Thread.sleep(1000);
    
    }
    

    }

    0 讨论(0)
  • 2021-02-01 09:23

    I had a very similar issue and the problem with destroy() not working was manifesting even with a single thread.

    Process process = processBuilder(ForeverRunningMain.class).start()
    long endTime = System.currentTimeMillis() + TIMEOUT_MS; 
    while (System.currentTimeMillis() < endTime) {
        sleep(50);
    }
    process.destroy();
    

    The process was not always destroyed if TIMEOUT_MS was too low. Adding an additional sleep() before destroy() fixed it (even though I don't have an explanation why):

    Thread.sleep(300);
    process.destroy();
    
    0 讨论(0)
  • 2021-02-01 09:27

    You should close the input/output/error streams to the process. We saw some issues in the past where the forked process was not completing properly due to those streams not being closed (even if they weren't being used).

    An exemplary solution:

    p.destroy();
    p.getInputStream().close();
    p.getOutputStream().close();
    p.getErrorStream().close();
    
    0 讨论(0)
  • 2021-02-01 09:30

    If subprocesses write anything to stdout or stderr (intentionally or not), that could cause trouble:

    "Because some native platforms only provide limited buffer size for standard input and output streams, failure to promptly write the input stream or read the output stream of the subprocess may cause the subprocess to block, and even deadlock."

    Source: http://www.javaworld.com/jw-12-2000/jw-1229-traps.html

    The whole article is IMO worth reading if you need to use Runtime.exec().

    0 讨论(0)
  • 2021-02-01 09:34

    There is a race condition between the time Runtime.exec kicks off a new thread to start a Process and when you tell that process to destroy itself.

    I'm on a linux machine so I will use the UNIXProcess.class file to illustrate.

    Runtime.exec(...) will create a new ProcessBuilder and start it which on a unix machine creates a new UNIXProcess instance. In the constructor of UNIXProcess there is this block of code which actually executes the process in a background (forked) thread:

    java.security.AccessController.doPrivileged(
                new java.security.PrivilegedAction() {
        public Object run() {
        Thread t = new Thread("process reaper") {
            public void run() {
                        try {
                            pid = forkAndExec(prog,
                          argBlock, argc,
                          envBlock, envc,
                          dir,
                          redirectErrorStream,
                          stdin_fd, stdout_fd, stderr_fd);
                        } catch (IOException e) {
                            gate.setException(e); /*remember to rethrow later*/
                            gate.exit();
                            return;
                        }
                        java.security.AccessController.doPrivileged(
                        new java.security.PrivilegedAction() {
                            public Object run() {
                            stdin_stream = new BufferedOutputStream(new
                                                    FileOutputStream(stdin_fd));
                            stdout_stream = new BufferedInputStream(new
                                                    FileInputStream(stdout_fd));
                            stderr_stream = new FileInputStream(stderr_fd);
                            return null;
                        }
                        });
                        gate.exit(); /* exit from constructor */
            int res = waitForProcessExit(pid);
            synchronized (UNIXProcess.this) {
                hasExited = true;
                exitcode = res;
                UNIXProcess.this.notifyAll();
            }
            }
        };
                t.setDaemon(true);
                t.start();
        return null;
        }
    });
    

    Notice that the background thread sets the field pid which is the UNIX process id. This will be used by destroy() to tell the OS which process to kill.

    Because there is no way to make sure that this background thread has run when destroy() is called, we may try to kill the process before it has run OR we may try to kill the process before pid field has been set; pid is uninitialized and therefore is 0. So I think calling destroy too early will do the equivalent of a kill -9 0

    There is even a comment in the UNIXProcess destroy() that alludes to this but only considers calling destroy after the process has already finished, not before it has started:

    // There is a risk that pid will be recycled, causing us to
    // kill the wrong process!  So we only terminate processes
    // that appear to still be running.  Even with this check,
    // there is an unavoidable race condition here, but the window
    // is very small, and OSes try hard to not recycle pids too
    // soon, so this is quite safe.
    

    The pid field is not even marked as volatile so we may not even see the most recent value all the time.

    0 讨论(0)
提交回复
热议问题