ProcessBuilder 介绍

Java 的 Process API 为开发者提供了执行操作系统命令的强大功能,但是某些 API 方法可能让你有些疑惑,没关系,这篇文章将详细介绍如何使用 ProcessBuilder API 来方便的操作系统命令。

ProcessBuilder 入门示例

我们通过演示如何调用 java -version 命令输出 JDK 版本号,来演示 ProcessBuilder 的入门用法。

  1. package com.wdbyte.os.process;
  2. import java.io.BufferedReader;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. import java.io.InputStreamReader;
  6. import org.apache.commons.io.IOUtils;
  7. /**
  8. * Process 输出Java 版本号
  9. * @author https://www.wdbyte.com
  10. */
  11. public class ProcessBuilderTest1 {
  12. public static void main(String[] args) throws IOException, InterruptedException {
  13. // 构建执行命令
  14. ProcessBuilder processBuilder = new ProcessBuilder("java","-version");
  15. // 重定向 ERROR 流(有些 JDK 版本 Java 命令通过 ERROR 流输出)
  16. processBuilder.redirectErrorStream(true);
  17. // 运行命令 java -version
  18. Process process = processBuilder.start();
  19. // 获取PID,这是一个 Java 9 方法
  20. long pid = process.pid();
  21. // 一次性获取运行结果
  22. String result = IOUtils.toString(process.getInputStream());
  23. // 等到运行结束
  24. int exitCode = process.waitFor();
  25. System.out.println("pid:" + pid);
  26. System.out.println("result:" + result);
  27. System.out.println("exitCode:" + exitCode);
  28. }
  29. }

在这段代码中,首先使用 ProcessBuilder 对象包装了要执行的命令 java -version,紧接着重定向 了要执行的进程的 ERROR 输出流(有些 JDK 版本 Java 命令通过 ERROR 流输出)。最后通过 start 方法执行命令,得到一个用于进程管理的 Process 对象,可以获取其 pid 和输出结果。

注意 IOUtils.toString(process.getInputStream());

这里使用了 commons-io 中的工具类把 InputStream 转为字符串。

commons-io Maven 依赖:

  1. <dependency>
  2. <groupId>commons-io</groupId>
  3. <artifactId>commons-io</artifactId>
  4. <version>2.12.0</version>
  5. </dependency>

运行得到输出:

  1. pid:80885
  2. result:java version "1.8.0_151"
  3. Java(TM) SE Runtime Environment (build 1.8.0_151-b12)
  4. Java HotSpot(TM) 64-Bit Server VM (build 25.151-b12, mixed mode)
  5. exitCode:0

ProcessBuilder 环境变量

在下面这个示例中,演示如何获取当前环境变量,以及如何修改环境变量并传入子进程中。

输出当前环境变量

  1. ProcessBuilder processBuilder = new ProcessBuilder();
  2. Map<String, String> environment = processBuilder.environment();
  3. environment.forEach((k, v) -> System.out.println(k + ":" + v));
  4. processBuilder.environment().put("my_website","www.wdbyte.com");

这会打印出当前所有环境变量。

  1. JAVA_HOME:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home
  2. COMMAND_MODE:unix2003
  3. JAVA_MAIN_CLASS_81717:com.wdbyte.os.process.ProcessBuilderTest2
  4. LOGNAME:darcy
  5. .....

添加一个环境变量

  1. processBuilder.environment().put("my_website","www.wdbyte.com");

打印出刚才添加的环境变量

  1. // Linux 或 MacOS 下 ,Windows 下无此命令
  2. processBuilder.command("/bin/bash", "-c", "echo $my_website");
  3. Process process = processBuilder.start();
  4. long pid = process.pid();
  5. String result = IOUtils.toString(process.getInputStream());
  6. int exitCode = process.waitFor();
  7. System.out.println("pid:" + pid);
  8. System.out.println("result:" + result);
  9. System.out.println("exitCode:" + exitCode);

这会输出:

  1. pid:81719
  2. result:www.wdbyte.com
  3. exitCode:0

ProcessBuilder 工作目录

使用 directory 方法可以修改子进程默认的工作目录,下面的示例中修改进程工作目录为 process 文件夹。

  1. package com.wdbyte.os.process;
  2. import java.io.File;
  3. import java.io.IOException;
  4. import org.apache.commons.io.IOUtils;
  5. /**
  6. * 修改工作目录
  7. * @author https://www.wdbyte.com
  8. */
  9. public class ProcessBuilderTest3 {
  10. private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process";
  11. public static void main(String[] args) throws IOException, InterruptedException {
  12. ProcessBuilder processBuilder = new ProcessBuilder();
  13. processBuilder.directory(new File(BASE_DIR));
  14. // /bin/bash 命令只在 linux or macos 下有效
  15. processBuilder.command("/bin/bash", "-c", "pwd");
  16. Process process = processBuilder.start();
  17. long pid = process.pid();
  18. String result = IOUtils.toString(process.getInputStream());
  19. int exitCode = process.waitFor();
  20. System.out.println("pid:" + pid);
  21. System.out.println("result:" + result);
  22. System.out.println("exitCode:" + exitCode);
  23. }
  24. }

输出:

  1. pid:82456
  2. result:/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process
  3. exitCode:0

ProcessBuilder I/O

在上面的示例中,都是把运行的新进程的输出通过 getInputStream 的方式读取到当前进程,然后输出,这种方式很不方便。日志输出常见的方式是输出到指定日志文件,ProcessBuilder 对此也有很好的支持。

输出到文件

使用 redirectOutput 可以指定日志输出的文件,这个方法会自动创建日志文件。下面的例子在指定目录下执行 ls-l 命令列出目录下的所有文件。

  1. package com.wdbyte.os.process;
  2. import java.io.File;
  3. import java.io.IOException;
  4. import java.nio.file.Files;
  5. /**
  6. * 输出日志到指定文件
  7. * @author https://www.wdbyte.com
  8. */
  9. public class ProcessBuilderTest4 {
  10. private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process";
  11. public static void main(String[] args) throws IOException, InterruptedException {
  12. ProcessBuilder processBuilder = new ProcessBuilder();
  13. processBuilder.directory(new File(BASE_DIR));
  14. processBuilder.command("/bin/bash", "-c", "ls -l");
  15. File logFile = new File(BASE_DIR + "/process_log.txt");
  16. // 输出到日志文件
  17. processBuilder.redirectOutput(logFile);
  18. // 追加日志到文件
  19. // processBuilder.redirectOutput(ProcessBuilder.Redirect.appendTo(logFile));
  20. // 是否输出ERROR日志到文件
  21. processBuilder.redirectErrorStream(true);
  22. Process process = processBuilder.start();
  23. long pid = process.pid();
  24. int exitCode = process.waitFor();
  25. System.out.println("pid:" + pid);
  26. System.out.println("exitCode:" + exitCode);
  27. // 读取日志文件
  28. Files.lines(logFile.toPath()).forEach(System.out::println);
  29. }
  30. }

输出日志:

  1. pid:30609
  2. exitCode:0
  3. total 96
  4. -rw-r--r-- 1 darcy staff 749 Jun 6 22:34 ExecDemo.java
  5. -rw-r--r-- 1 darcy staff 445 Jun 7 14:59 ExecDemo2.java
  6. -rw-r--r-- 1 darcy staff 2011 Jun 7 15:33 ProcessBuilder10.java
  7. -rw-r--r-- 1 darcy staff 1807 Jun 6 22:54 ProcessBuilderTest1.java
  8. -rw-r--r-- 1 darcy staff 1054 Jun 6 23:01 ProcessBuilderTest2.java
  9. -rw-r--r-- 1 darcy staff 963 Jun 6 23:05 ProcessBuilderTest3.java
  10. -rw-r--r-- 1 darcy staff 1295 Jun 7 17:02 ProcessBuilderTest4.java
  11. -rw-r--r-- 1 darcy staff 1250 Jun 6 22:34 ProcessBuilderTest5.java
  12. -rw-r--r-- 1 darcy staff 929 Jun 6 22:34 ProcessBuilderTest6.java
  13. -rw-r--r-- 1 darcy staff 911 Jun 6 22:34 ProcessBuilderTest7.java
  14. -rw-r--r-- 1 darcy staff 1305 Jun 6 22:34 ProcessBuilderTest8.java
  15. -rw-r--r-- 1 darcy staff 1278 Jun 7 14:59 ProcessBuilderTest9.java
  16. -rw-r--r-- 1 darcy staff 0 Jun 7 17:03 process_log.txt

如果想要追加日志到指定文件,应该使用:

  1. processBuilder.redirectOutput(ProcessBuilder.Redirect.appendTo(logFile));

使用 processBuilder 也可以指定 INFOERROR 日志到不同的文件。

  1. ProcessBuilder processBuilder = new ProcessBuilder();
  2. processBuilder.directory(new File(BASE_DIR));
  3. // 执行命令 xxx,命令不存在,会报 ERROR 日志
  4. processBuilder.command("/bin/bash", "-c", "xxx");
  5. File infoLogFile = new File(BASE_DIR + "/process_log_info.txt");
  6. File errorLogFile = new File(BASE_DIR + "/process_log_error.txt");
  7. // 日志输出到文件
  8. processBuilder.redirectOutput(infoLogFile);
  9. processBuilder.redirectError(errorLogFile);
  10. Process process = processBuilder.start();
  11. // 读取 ERROR 日志
  12. Files.lines(errorLogFile.toPath()).forEach(System.out::println);

运行输出:

  1. /bin/bash: xxx: command not found

输出到当前进程

在这个示例中,将看到 inheritIO() 方法的作用。当我们想将子进程的 I/O 重定向到当前进程的标准 I/O 时,可以使用这个方法:

  1. package com.wdbyte.os.process;
  2. import java.io.File;
  3. import java.io.IOException;
  4. /**
  5. * 子线程 I/O 重定向到当前线程
  6. * @author https://www.wdbyte.com
  7. */
  8. public class ProcessBuilderTest6 {
  9. public static void main(String[] args) throws IOException, InterruptedException {
  10. ProcessBuilder processBuilder = new ProcessBuilder();
  11. processBuilder.directory(new File("./"));
  12. processBuilder.command("/bin/bash", "-c", "ls -l");
  13. // 把子线程 I/O 输出重定向当前进程
  14. processBuilder.inheritIO();
  15. Process process = processBuilder.start();
  16. int exitCode = process.waitFor();
  17. System.out.println("exitCode:" + exitCode);
  18. }
  19. }

这会输出:

  1. total 2904
  2. -rw-r--r-- 1 darcy staff 5822 May 2 22:33 ArrayList.uml
  3. -rw-r--r-- 1 darcy staff 16555 May 16 16:07 README.md
  4. -rw-r--r-- 1 darcy staff 333 May 4 19:30 core-java-20.iml
  5. drwxr-xr-x 16 darcy staff 512 Jun 2 22:03 core-java-modules
  6. exitCode:0

在这个示例中,通过使用inheritIO()方法,我们在 IDE 的控制台中看到了一个简单命令结果的输出。

ProcessBuilder 管道操作

从 Java 9 开始,ProcessBuilder 引入了管道概念,可以把一个进程的输出作为另一个进程的输入再次操作。

  1. public static List<Process> startPipeline(List<ProcessBuilder> builders)

使用这个方法我们可以进行如这样的常见操作:ls -l | wc -l

ls -l | wc -l :列出文件目录,然后统计输出的行数。

下面演示如何使用 startPipeline.

  1. package com.wdbyte.os.process;
  2. import java.io.File;
  3. import java.io.IOException;
  4. import java.lang.ProcessBuilder.Redirect;
  5. import java.nio.file.Files;
  6. import java.util.Arrays;
  7. import java.util.List;
  8. /**
  9. * Java 9 中新增的管道操作
  10. * @author https://www.wdbyte.com
  11. */
  12. public class ProcessBuilderTest8 {
  13. private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process";
  14. public static void main(String[] args) throws IOException, InterruptedException {
  15. ProcessBuilder ls = new ProcessBuilder("/bin/bash", "-c", "ls -l");
  16. ProcessBuilder wc = new ProcessBuilder("wc", "-l");
  17. // 追加日志到文件
  18. File pipeLineLogFile = getFile(BASE_DIR + "/pipe_line_log.txt");
  19. wc.redirectOutput(Redirect.appendTo(pipeLineLogFile));
  20. List<Process> processes = ProcessBuilder.startPipeline(Arrays.asList(ls, wc));
  21. Process process = processes.get(processes.size() - 1);
  22. System.out.println("pid:" + process.pid());
  23. System.out.println("exitCode:" + process.waitFor());
  24. Files.lines(pipeLineLogFile.toPath()).forEach(System.out::println);
  25. }
  26. public static File getFile(String filePath) throws IOException {
  27. File logFile = new File(filePath);
  28. if (!logFile.exists()) {
  29. logFile.createNewFile();
  30. }
  31. return logFile;
  32. }
  33. }

这会输出:

  1. pid:33518
  2. exitCode:0
  3. 21

ProcessBuilder 超时与终止

进程有时不能按照自己想要的情况运行,需要对进程进行管理,常见的操作是超时控制以及进程退出。下面通过一个例子来演示如何操作。

先编译一个用于测试的 Java 类 ExecDemo.java,此类每隔一秒输出一个数字,共输出10个数字,预计需要10s输出完毕。

下面是代码部分:

  1. import java.io.IOException;
  2. /**
  3. * @author https://www.wdbyte.com
  4. */
  5. public class ExecDemo {
  6. public static void main(String[] args) throws InterruptedException {
  7. System.out.println("开始处理数据...");
  8. for (int i = 0; i < 10; i++) {
  9. Thread.sleep(1000);
  10. System.out.println(i);
  11. }
  12. System.out.println("数据处理完毕");
  13. }
  14. }

再编写一个 ProcessBuilder 来执行 ExceDemo,但是在执行 3 秒后就判断是否运行完成,如果没有则杀死进程。

  1. package com.wdbyte.os.process;
  2. import java.io.File;
  3. import java.io.IOException;
  4. import java.util.concurrent.TimeUnit;
  5. /**
  6. * 运行一个 Java 程序
  7. * 等待一定时间后检查状态,未结束则直接杀死进程。
  8. *
  9. * @author https://www.wdbyte.com
  10. */
  11. public class ProcessBuilderTest9 {
  12. private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process";
  13. public static void main(String[] args) throws IOException, InterruptedException {
  14. ProcessBuilder processBuilder = new ProcessBuilder();
  15. processBuilder.directory(new File(BASE_DIR));
  16. processBuilder.command("java", "ExecDemo.java");
  17. // 把子线程 I/O 输出重定向当前进程
  18. processBuilder.inheritIO();
  19. Process process = processBuilder.start();
  20. // 等待一定时间
  21. boolean waitFor = process.waitFor(3, TimeUnit.SECONDS);
  22. System.out.println("waitFor:" + waitFor);
  23. // 若未退出,杀死子进程
  24. if (!waitFor) {
  25. process.destroyForcibly();
  26. process.waitFor();
  27. System.out.println("杀死进程:" + process);
  28. }
  29. }
  30. }

这会输出:

  1. 开始处理数据...
  2. 0
  3. 1
  4. waitFor:false
  5. 杀死进程:Process[pid=35084, exitValue=137]

在这段代码中,destroyForcibly() 用于杀死进程,但是杀死进程并不是瞬间完成的,所以接着使用 waitFor() 来等待程序真正被杀死退出。

ProcessBuilder 异步处理

很多情况下,在执行一个命令启动一个新线程后,我们不想阻塞等待进程的完成,想要异步化,在进程执行完成后进行通知回调。这时可以使用 CompletableFuture 来实现这个功能。

  1. package com.wdbyte.os.process;
  2. import java.io.File;
  3. import java.io.IOException;
  4. import java.util.concurrent.CompletableFuture;
  5. /**
  6. * @author https://www.wdbyte.com
  7. */
  8. public class ProcessBuilderTest10 {
  9. private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process";
  10. public static void main(String[] args) throws InterruptedException {
  11. ProcessBuilder processBuilder = new ProcessBuilder();
  12. processBuilder.directory(new File(BASE_DIR));
  13. processBuilder.command("java", "ExecDemo.java");
  14. // 把子线程 I/O 输出重定向当前进程
  15. processBuilder.inheritIO();
  16. // 创建 CompletableFuture 对象
  17. CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
  18. try {
  19. // 命令执行
  20. Process process = processBuilder.start();
  21. // 任务超时时间
  22. process.waitFor();
  23. } catch (IOException e) {
  24. throw new RuntimeException(e);
  25. } catch (InterruptedException e) {
  26. throw new RuntimeException(e);
  27. }
  28. return null;
  29. });
  30. // 注册回调函数,处理异步等待的结果
  31. future.thenAccept(result -> {
  32. System.out.println("进程执行结束");
  33. });
  34. System.out.println("主进程等待");
  35. Thread.sleep(20 * 1000);
  36. }
  37. }

这会输出:

  1. 主进程等待
  2. 开始处理数据...
  3. 0
  4. 1
  5. 2
  6. 3
  7. 4
  8. 5
  9. 6
  10. 7
  11. 8
  12. 9
  13. 数据处理完毕
  14. 进程执行结束

ProcessBuilder 总结

在这篇文章中,我们详细介绍了 ProcessBuilder 的具体用法,并且给出了常用的操作示例。同时也介绍了 Java 9 开始为 ProcessBuilder 引入的管道操作,最后介绍如何对 Process 进程进行异步处理。

一如既往,文章中代码存放在 Github.com/niumoo/javaNotes.

本文原发于网站:https://www.wdbyte.com/java/os/processbuilder/

我的公众号:ProcessBuilder API 使用教程

使用 ProcessBuilder API 优化你的流程的更多相关文章

  1. 【百度地图API】今日小年大进步,齐头共进贺佳节——API优化升级上线,不再增加内存消耗

    原文:[百度地图API]今日小年大进步,齐头共进贺佳节--API优化升级上线,不再增加内存消耗 任务描述: 今天是2011年01月26日,小年夜.百度地图API在小年夜献给广大API爱好者一份给力的礼 ...

  2. 工作流JBPM_day01:3-使用JBPM的API添加与执行流程

    工作流JBPM_day01:3-使用JBPM的API添加与执行流程 流程定义画完得到压缩文件--->部署流程定义-->启动流程实例-->查询我的个人任务列表-->办理任务--& ...

  3. 第六节:框架搭建之EF的Fluent Api模式的使用流程

    一. 前言 沉寂了约一个月的时间,今天用一篇简单的文章重新回归博客,主要来探讨一下Fluent Api模式在实际项目中的使用流程. 1. Fluent API属于EF CodeFirst模式的一种,E ...

  4. 数据筛选和API优化

    筛选数据 需求:如果数据库中存在OrderNum相同,且IsDefault不同的记录,那么IsDefault值为0的记录将替换值为1的记录(IsDefault值为1的记录不展示). 由于查出来的数据不 ...

  5. 使用flowable 6.1.2 REST API 运行请假审批流程

    一.下载 flowable rest war 包 http://download.csdn.net/detail/teamlet/9913312 二.部署 复制flowable REST.war到To ...

  6. AS3 巧用事件api简化鼠标拖动流程

     拖动,按照一般人的定义,拖动就是鼠标按下的时候移动鼠标,这里面有三个过程,分别是按下.移动鼠标和弹起.以stage为例,大家的实现步骤通常如下:(PS:此处不讨论startDrag和stopDrag ...

  7. CMDB学习之六 --客户端请求测试,服务端api优化

    客户端使用agent 请求测试,agent使用的POST 请求,使用requests模块 本地采集,汇报服务端 #!/usr/bin/env python # -*- coding:utf-8 -*- ...

  8. 项目API接口鉴权流程总结

    权益需求对接中,公司跟第三方公司合作,有时我们可能作为甲方,提供接口给对方,有时我们也作为乙方,调对方接口,这就需要API使用签名方法(Sign)对接口进行鉴权.每一次请求都需要在请求中包含签名信息, ...

  9. HTML5的classList API优化对样式名className的操作

    //添加一个class elem.classList.add(classname); //删除一个class elem.classList.remove(classname); //判断一个class ...

  10. Mali GPU OpenGL ES 应用性能优化--測试+定位+优化流程

    1. 使用DS-5 Streamline定位瓶颈 DS-5 Streamline要求GPU驱动启用性能測试,在Mali GPU驱动中激活性能測试对性能影响微不足道. 1.1 DS-5 Streamli ...

随机推荐

  1. dart基础---->函数传值

    1. string type main(List<String> args) { String name = "huhx"; changIt(name); print( ...

  2. 利用Karlibr生成April标定板图像

    1 关键的命令 rosrun kalibr kalibr_create_target_pdf --type apriltag --nx 6 --ny 6 --tsize 0.02 --tspace 0 ...

  3. NetCore 使用 Swashbuckle 搭建 SwaggerHub

    什么是SwaggerHub? Hub 谓之 中心, 所以 SwaggerHub即swagger中心. 什么时候需要它? 通常, 公司都拥有多个服务, 例如商品服务, 订单服务, 用户服务, 等等, 每 ...

  4. pandas之分类操作

    通常情况下,数据集中会存在许多同一类别的信息,比如相同国家.相同行政编码.相同性别等,当这些相同类别的数据多次出现时,就会给数据处理增添许多麻烦,导致数据集变得臃肿,不能直观.清晰地展示数据. 针对上 ...

  5. python:selenium爬取boss网站被关小黑屋

    问题描述:使用selenium访问次数过多,被boss反爬封掉IP,这种方式有什么好一点的解决方法,首次可以用图形验证解封,今天访问次数过多,被关进了小黑屋 首次让我用图形界面解封 不过还好,手动解封 ...

  6. React redux toolkit: Uncaught Error:[Immer] An immer producer returned a new...

    React在写一个购物车的redux toolkit时遇到了问题.核心代码如下: import { createSlice } from "@reduxjs/toolkit"; c ...

  7. 指针和引用(pointer and reference),传值和传址

    pass by adress pass by reference和pass by pointer的共同点都在于传址,都是对于对象的地址的复制,而不会对对象进行产生副本的操作. pass by refe ...

  8. 一个.Net版本的ChatGPT SDK

    ChatGPT大火,用它来写代码.写表白书.写文章.写对联.写报告.写周边...... 啥都会! 个人.小公司没有能力开发大模型,但基于开放平台,根据特定的场景开发应用,却是非常火热的. 为了避免重复 ...

  9. css实现文本溢出省略号

    CSS常用属性: overflow:hidden; //超出的文本隐藏 text-overflow:ellipsis; //溢出用省略号显示 white-space:nowrap; //溢出不换行,只 ...

  10. ElementPlus 组件全局配置

    友链:语雀,在线文档协同平台 官方提供的全局配置:Config Provider 本文只做简单的模板参考,具体的配置请根据自己的业务灵活设置,如果你使用的是其它的ui框架,原理应该都差不多 入口文件的 ...