十、Java 9 新特性 – 改进进程管理 API

Java 9 这个版本对进程管理方面的改进也是相当大的。在为数不多的几次 Java 项目中,有偶尔用到多线程,但对多进程和进程方面的了解还真是太少。

我想,大部分人应该跟我一样,在编程之外知道有进程的东西的存在,在 Java 中反而会忽视,因为多线程和并发 ( Concurrency ) 的存在感更强吧。

这次Java 9 对进程管理的改进主要是提供了 ProcessHandle

ProcessHandle 类

该类在java.lang 包中,且处于 java.base 模块中。

ProcessHandle 可以用于获取进程信息,监听和检查进程的状态,并且可以监听进程的退出

主要提供了以下几个方法

方法 说明
static allProcesses() 返回当前进程可见的所有进程的快照
static current() 返回当前进程的 ProcessHandle 实例
static of(long pid) 返回现有本机进程的 Optional <ProcessHandle>
children() 返回进程的当前直接子进程的快照
compareTo(ProcessHandle other) 比较两个进程
descendants() 返回当前进程后代的快照
destroy() 请求杀死当前进程
destroyForcibly() 强制杀死该进程
equals(Object other) 如果 other 对象为非 null,且具有相同的实现,并且表示相同的系统进程,则返回 true; 否则返回 false
hashCode() 返回此 ProcessHandle 的哈希值
info() 返回有关该进程的信息的快照
isAlive() 测试此 ProcessHandle 表示的进程是否处于活动状态
onExit() 当进程终止时返回 CompletableFuture <ProcessHandle>
parent() 返回当前进程的父进程 Optional<ProcessHandle> ,因为当前进程可能是初始进程,所以父进程不一定存在
pid() 返回当前进程的系统进程的 id
supportsNormalTermination() 如果 destroy() 正常终止进程,则返回 true

ProcessHandle 类用于标识并提供对 native 进程的控制,可以监控每个单独的进程的活跃度,列出其子进程 ( 线程 ) ,获取有关进程的信息或将其销毁。

而很早就存在的 Process 类,它的实例由当前进程启动,只提供了对进程输入,输出和错误流的访问

native 进程 ID 是操作系统分配给进程的标识号,这个 ID 值的范围取决于操作系统,例如,嵌入式系统可能使用 16 位值。

关于这个 ProcessHandle 类,还有很多信息,如果你对此感兴趣,可以访问 官方 API 文档: Interface ProcessHandle

范例

我们写一个范例来演示下 ProcessHandle 如何使用,在当前工作区创建一个文件 ProcessHandleTester.java,并输入以下内容

import java.time.ZoneId;
import java.util.stream.Stream;
import java.util.stream.Collectors;
import java.io.IOException;
public class ProcessHandleTester {
   public static void main(String[] args) throws IOException {
      // Windows 有效
      ProcessBuilder pb = new ProcessBuilder("notepad.exe");
      String np = "Not Present";
      Process p = pb.start();
      ProcessHandle.Info info = p.info();
      System.out.printf("Process ID : %s%n", p.pid());
      System.out.printf("Command name : %s%n", info.command().orElse(np));
      System.out.printf("Command line : %s%n", info.commandLine().orElse(np));
      System.out.printf("Start time: %s%n",
         info.startInstant().map(i -> i.atZone(ZoneId.systemDefault())
         .toLocalDateTime().toString()).orElse(np));
      System.out.printf("Arguments : %s%n",
         info.arguments().map(a -> Stream.of(a).collect(
         Collectors.joining(" "))).orElse(np));
      System.out.printf("User : %s%n", info.user().orElse(np));
   } 

运行该范例,输出结果如下

[penglei@liyoudong.top java9]$ javac ProcessHandleTester.java && java ProcessHandleTester
Process ID : 5800
Command name : C:\Windows\System32\notepad.exe
Command line : Not Present
Start time: 2017-11-04T21:35:03.626
Arguments : Not Present
User: administrator