ProcessHandle onExit 有空数据
ProcessHandle onExit has empty data
我目前正在测试来自 java 9 的进程 API,我在使用以下代码时遇到了一些问题:
Process process = new ProcessBuilder( List.of("ping", "-i", "1", "-c", "4", "google.com")).start();
CompletableFuture<Void> startTimeFuture = process.toHandle().onExit()
.thenApply(ProcessHandle::info)
.thenApply(ProcessHandle.Info::startInstant)
.thenAccept(System.out::println);
startTimeFuture.get();
当我执行此代码段时,我在终端中得到 Optional.empty。
Javadoc 指出 info
方法 returns 任何可用的数据,因此我怀疑 JVM 无法获取有关衍生进程的信息。但是,当我将来尝试从 ProcessHandle
获取 pid 时,我会得到适当的值。
综上所述,我的问题:
调用onExit()
后有没有办法得到非空ProcessHandle.Info
?
我正在使用 Ubuntu 16.04 LTS
Edit - 这是我执行 ping -i 1 -c 5 google.com
时终端的输出
PING google.com (xxx.xxx.16.46) 56(84) bytes of data.
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=1 ttl=52 time=6.71 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=2 ttl=52 time=6.26 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=3 ttl=52 time=16.6 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=4 ttl=52 time=10.6 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=5 ttl=52 time=13.4 ms
--- google.com ping statistics ---
5 packets transmitted, 5 received, 0% packet loss, time 4007ms
rtt min/avg/max/mdev = 6.267/10.746/16.667/3.968 ms
更新的用例:- 我想检查是否可以,给定命令的时间是多少执行,例如通过调用 ProcessHandle.Info::totalCpuDuration
我想我找到了这种行为的原因(至少在 linux 分布上)。
ProcessHandle.Info
对象是使用以下方法创建的:
public static ProcessHandle.Info info(long pid, long startTime) {
Info info = new Info();
info.info0(pid);
if (startTime != info.startTime) {
info.command = null;
info.arguments = null;
info.startTime = -1L;
info.totalTime = -1L;
info.user = null;
}
return info;
}
其中 info.info0(pid)
是对本机方法的调用。
所以我下载了openjdk源代码并检查了这个方法的实现。在 linux 上,JVM 通过读取 /proc/{pid}/stat
、/proc/{pid}/cmdline
、/proc/{pid}/exe
来检索进程数据,这些数据在进程终止后不再可用。
回答我的问题:
无法获得已完成进程的 ProcessHandle.Info
。
我目前正在测试来自 java 9 的进程 API,我在使用以下代码时遇到了一些问题:
Process process = new ProcessBuilder( List.of("ping", "-i", "1", "-c", "4", "google.com")).start();
CompletableFuture<Void> startTimeFuture = process.toHandle().onExit()
.thenApply(ProcessHandle::info)
.thenApply(ProcessHandle.Info::startInstant)
.thenAccept(System.out::println);
startTimeFuture.get();
当我执行此代码段时,我在终端中得到 Optional.empty。
Javadoc 指出 info
方法 returns 任何可用的数据,因此我怀疑 JVM 无法获取有关衍生进程的信息。但是,当我将来尝试从 ProcessHandle
获取 pid 时,我会得到适当的值。
综上所述,我的问题:
调用onExit()
后有没有办法得到非空ProcessHandle.Info
?
我正在使用 Ubuntu 16.04 LTS
Edit - 这是我执行 ping -i 1 -c 5 google.com
PING google.com (xxx.xxx.16.46) 56(84) bytes of data.
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=1 ttl=52 time=6.71 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=2 ttl=52 time=6.26 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=3 ttl=52 time=16.6 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=4 ttl=52 time=10.6 ms
64 bytes from waw02s14-in-f14.1e100.net (xxx.xxx.16.46): icmp_seq=5 ttl=52 time=13.4 ms
--- google.com ping statistics ---
5 packets transmitted, 5 received, 0% packet loss, time 4007ms rtt min/avg/max/mdev = 6.267/10.746/16.667/3.968 ms
更新的用例:- 我想检查是否可以,给定命令的时间是多少执行,例如通过调用 ProcessHandle.Info::totalCpuDuration
我想我找到了这种行为的原因(至少在 linux 分布上)。
ProcessHandle.Info
对象是使用以下方法创建的:
public static ProcessHandle.Info info(long pid, long startTime) {
Info info = new Info();
info.info0(pid);
if (startTime != info.startTime) {
info.command = null;
info.arguments = null;
info.startTime = -1L;
info.totalTime = -1L;
info.user = null;
}
return info;
}
其中 info.info0(pid)
是对本机方法的调用。
所以我下载了openjdk源代码并检查了这个方法的实现。在 linux 上,JVM 通过读取 /proc/{pid}/stat
、/proc/{pid}/cmdline
、/proc/{pid}/exe
来检索进程数据,这些数据在进程终止后不再可用。
回答我的问题:
无法获得已完成进程的 ProcessHandle.Info
。