为什么当子进程仍处于打开状态时,Java 进程会从 Gradle 挂起?

2022-09-04 04:36:22

如果在 java 中创建的进程创建了一个子进程,但随后返回,则 JVM 将挂起,但没有进程 ID。

下面的示例应用程序(需要 Windows 和 Java 7)

import java.io.File;
import java.io.IOException;
import java.lang.ProcessBuilder.Redirect;
import java.nio.file.Files;

public class SubProcessHang {

    public static void main(String[] args) throws IOException, InterruptedException {
        ProcessBuilder builder = new ProcessBuilder("cmd", "/c", "start", "notepad.exe");
        File output = Files.createTempFile("output", "txt").toFile();
        builder.redirectError(Redirect.to(output));
        builder.redirectOutput(Redirect.to(output));
        Process process = builder.start();
        process.waitFor();
        int exitValue = process.exitValue();
        System.out.println("Process exit value:: " + exitValue);
        System.out.println("Output file length:: " + output.length());
        System.exit(exitValue);
    }
}

当应用程序运行时,它会创建三个进程:java --> cmd -->记事本cmd立即返回,java调用System.exit(0),这会杀死java进程。但是记事本仍然存在,并且,当从gradle(或eclipse)运行时,JVM会一直挂起,直到该进程消失,而不会返回它的返回值。

所以子进程还活着,但亲子进程已经被部分杀死,但现在永远搁浅了。

用于重现此内容的 build.gradle 脚本

apply plugin: 'java'
apply plugin: 'application'
mainClassName = "SubProcessHang"

执行“gradle run”并获取以下输出:

C:\HangDemo>gradlew run
:compileJava
:processResources UP-TO-DATE
:classes
:run
Process exit value:: 0
Output file length:: 0
> Building 75% > :run

我知道这一定与Java进程的创建方式有关,但我不知道该怎么做。

除了获取正在运行的java进程的ID并在关闭钩子中杀死所有子进程之外,我还能做些什么?


答案 1

流程的文档说

默认情况下,创建的子流程没有自己的终端或控制台。它的所有标准 I/O(即 stdin、stdout、stderr)操作都将重定向到父进程,在那里可以通过使用 getOutputStream()、getInputStream() 和 getErrorStream() 方法获取的流来访问它们。父进程使用这些流将输入馈送到子进程并从子进程获取输出。由于某些本机平台仅为标准输入和输出流提供有限的缓冲区大小,因此未能及时写入输入流或读取子进程的输出流可能会导致子进程阻塞,甚至死锁。

http://docs.oracle.com/javase/7/docs/api/java/lang/Process.html

也许您的过程正在创建标准输出或标准输出。尝试排空输入流和错误流。


答案 2

我想说这个答案可能有助于获取子进程ID和这个 - 在Windows环境中杀死它们。

希望有所帮助!


推荐