使用 ProcessBuilder API 优化你的流程
2023-06-14 09:20:51来源:博客园
Java 的 Process API 为开发者提供了执行操作系统命令的强大功能,但是某些 API 方法可能让你有些疑惑,没关系,这篇文章将详细介绍如何使用 ProcessBuilder API 来方便的操作系统命令。
ProcessBuilder 入门示例我们通过演示如何调用 java -version
命令输出 JDK 版本号,来演示 ProcessBuilder
的入门用法。
package com.wdbyte.os.process;import java.io.BufferedReader;import java.io.IOException;import java.io.InputStream;import java.io.InputStreamReader;import org.apache.commons.io.IOUtils;/** * Process 输出Java 版本号 * @author https://www.wdbyte.com */public class ProcessBuilderTest1 { public static void main(String[] args) throws IOException, InterruptedException { // 构建执行命令 ProcessBuilder processBuilder = new ProcessBuilder("java","-version"); // 重定向 ERROR 流(有些 JDK 版本 Java 命令通过 ERROR 流输出) processBuilder.redirectErrorStream(true); // 运行命令 java -version Process process = processBuilder.start(); // 获取PID,这是一个 Java 9 方法 long pid = process.pid(); // 一次性获取运行结果 String result = IOUtils.toString(process.getInputStream()); // 等到运行结束 int exitCode = process.waitFor(); System.out.println("pid:" + pid); System.out.println("result:" + result); System.out.println("exitCode:" + exitCode); }}
在这段代码中,首先使用 ProcessBuilder 对象包装了要执行的命令 java -version
,紧接着重定向 了要执行的进程的 ERROR 输出流(有些 JDK 版本 Java 命令通过 ERROR 流输出)。最后通过 start
方法执行命令,得到一个用于进程管理的 Process
对象,可以获取其 pid
和输出结果。
(资料图片仅供参考)
注意
IOUtils.toString(process.getInputStream());
这里使用了 commons-io 中的工具类把 InputStream 转为字符串。
commons-io
Maven 依赖:
commons-io commons-io 2.12.0
运行得到输出:
pid:80885result:java version "1.8.0_151"Java(TM) SE Runtime Environment (build 1.8.0_151-b12)Java HotSpot(TM) 64-Bit Server VM (build 25.151-b12, mixed mode)exitCode:0
ProcessBuilder 环境变量在下面这个示例中,演示如何获取当前环境变量,以及如何修改环境变量并传入子进程中。
输出当前环境变量。
ProcessBuilder processBuilder = new ProcessBuilder();Map environment = processBuilder.environment();environment.forEach((k, v) -> System.out.println(k + ":" + v));processBuilder.environment().put("my_website","www.wdbyte.com");
这会打印出当前所有环境变量。
JAVA_HOME:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/HomeCOMMAND_MODE:unix2003JAVA_MAIN_CLASS_81717:com.wdbyte.os.process.ProcessBuilderTest2LOGNAME:darcy.....
添加一个环境变量。
processBuilder.environment().put("my_website","www.wdbyte.com");
打印出刚才添加的环境变量。
// Linux 或 MacOS 下 ,Windows 下无此命令processBuilder.command("/bin/bash", "-c", "echo $my_website");Process process = processBuilder.start();long pid = process.pid();String result = IOUtils.toString(process.getInputStream());int exitCode = process.waitFor();System.out.println("pid:" + pid);System.out.println("result:" + result);System.out.println("exitCode:" + exitCode);
这会输出:
pid:81719result:www.wdbyte.comexitCode:0
ProcessBuilder 工作目录使用 directory
方法可以修改子进程默认的工作目录,下面的示例中修改进程工作目录为 process
文件夹。
package com.wdbyte.os.process;import java.io.File;import java.io.IOException;import org.apache.commons.io.IOUtils;/** * 修改工作目录 * @author https://www.wdbyte.com */public class ProcessBuilderTest3 { private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process"; public static void main(String[] args) throws IOException, InterruptedException { ProcessBuilder processBuilder = new ProcessBuilder(); processBuilder.directory(new File(BASE_DIR)); // /bin/bash 命令只在 linux or macos 下有效 processBuilder.command("/bin/bash", "-c", "pwd"); Process process = processBuilder.start(); long pid = process.pid(); String result = IOUtils.toString(process.getInputStream()); int exitCode = process.waitFor(); System.out.println("pid:" + pid); System.out.println("result:" + result); System.out.println("exitCode:" + exitCode); }}
输出:
pid:82456result:/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/processexitCode:0
ProcessBuilder I/O在上面的示例中,都是把运行的新进程的输出通过 getInputStream
的方式读取到当前进程,然后输出,这种方式很不方便。日志输出常见的方式是输出到指定日志文件,ProcessBuilder
对此也有很好的支持。
使用 redirectOutput
可以指定日志输出的文件,这个方法会自动创建日志文件。下面的例子在指定目录下执行 ls-l
命令列出目录下的所有文件。
package com.wdbyte.os.process;import java.io.File;import java.io.IOException;import java.nio.file.Files;/** * 输出日志到指定文件 * @author https://www.wdbyte.com */public class ProcessBuilderTest4 { private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process"; public static void main(String[] args) throws IOException, InterruptedException { ProcessBuilder processBuilder = new ProcessBuilder(); processBuilder.directory(new File(BASE_DIR)); processBuilder.command("/bin/bash", "-c", "ls -l"); File logFile = new File(BASE_DIR + "/process_log.txt"); // 输出到日志文件 processBuilder.redirectOutput(logFile); // 追加日志到文件 // processBuilder.redirectOutput(ProcessBuilder.Redirect.appendTo(logFile)); // 是否输出ERROR日志到文件 processBuilder.redirectErrorStream(true); Process process = processBuilder.start(); long pid = process.pid(); int exitCode = process.waitFor(); System.out.println("pid:" + pid); System.out.println("exitCode:" + exitCode); // 读取日志文件 Files.lines(logFile.toPath()).forEach(System.out::println); }}
输出日志:
pid:30609exitCode:0total 96-rw-r--r-- 1 darcy staff 749 Jun 6 22:34 ExecDemo.java-rw-r--r-- 1 darcy staff 445 Jun 7 14:59 ExecDemo2.java-rw-r--r-- 1 darcy staff 2011 Jun 7 15:33 ProcessBuilder10.java-rw-r--r-- 1 darcy staff 1807 Jun 6 22:54 ProcessBuilderTest1.java-rw-r--r-- 1 darcy staff 1054 Jun 6 23:01 ProcessBuilderTest2.java-rw-r--r-- 1 darcy staff 963 Jun 6 23:05 ProcessBuilderTest3.java-rw-r--r-- 1 darcy staff 1295 Jun 7 17:02 ProcessBuilderTest4.java-rw-r--r-- 1 darcy staff 1250 Jun 6 22:34 ProcessBuilderTest5.java-rw-r--r-- 1 darcy staff 929 Jun 6 22:34 ProcessBuilderTest6.java-rw-r--r-- 1 darcy staff 911 Jun 6 22:34 ProcessBuilderTest7.java-rw-r--r-- 1 darcy staff 1305 Jun 6 22:34 ProcessBuilderTest8.java-rw-r--r-- 1 darcy staff 1278 Jun 7 14:59 ProcessBuilderTest9.java-rw-r--r-- 1 darcy staff 0 Jun 7 17:03 process_log.txt
如果想要追加日志到指定文件,应该使用:
processBuilder.redirectOutput(ProcessBuilder.Redirect.appendTo(logFile));
使用 processBuilder
也可以指定 INFO
和 ERROR
日志到不同的文件。
ProcessBuilder processBuilder = new ProcessBuilder();processBuilder.directory(new File(BASE_DIR));// 执行命令 xxx,命令不存在,会报 ERROR 日志processBuilder.command("/bin/bash", "-c", "xxx");File infoLogFile = new File(BASE_DIR + "/process_log_info.txt");File errorLogFile = new File(BASE_DIR + "/process_log_error.txt");// 日志输出到文件processBuilder.redirectOutput(infoLogFile);processBuilder.redirectError(errorLogFile);Process process = processBuilder.start();// 读取 ERROR 日志Files.lines(errorLogFile.toPath()).forEach(System.out::println);
运行输出:
/bin/bash: xxx: command not found
输出到当前进程在这个示例中,将看到 inheritIO()
方法的作用。当我们想将子进程的 I/O 重定向到当前进程的标准 I/O 时,可以使用这个方法:
package com.wdbyte.os.process;import java.io.File;import java.io.IOException;/** * 子线程 I/O 重定向到当前线程 * @author https://www.wdbyte.com */public class ProcessBuilderTest6 { public static void main(String[] args) throws IOException, InterruptedException { ProcessBuilder processBuilder = new ProcessBuilder(); processBuilder.directory(new File("./")); processBuilder.command("/bin/bash", "-c", "ls -l"); // 把子线程 I/O 输出重定向当前进程 processBuilder.inheritIO(); Process process = processBuilder.start(); int exitCode = process.waitFor(); System.out.println("exitCode:" + exitCode); }}
这会输出:
total 2904-rw-r--r-- 1 darcy staff 5822 May 2 22:33 ArrayList.uml-rw-r--r-- 1 darcy staff 16555 May 16 16:07 README.md-rw-r--r-- 1 darcy staff 333 May 4 19:30 core-java-20.imldrwxr-xr-x 16 darcy staff 512 Jun 2 22:03 core-java-modulesexitCode:0
在这个示例中,通过使用inheritIO()方法,我们在 IDE 的控制台中看到了一个简单命令结果的输出。
ProcessBuilder 管道操作从 Java 9 开始,ProcessBuilder 引入了管道概念,可以把一个进程的输出作为另一个进程的输入再次操作。
public static List startPipeline(List builders)
使用这个方法我们可以进行如这样的常见操作:ls -l | wc -l
ls -l | wc -l
:列出文件目录,然后统计输出的行数。
下面演示如何使用 startPipeline
.
package com.wdbyte.os.process;import java.io.File;import java.io.IOException;import java.lang.ProcessBuilder.Redirect;import java.nio.file.Files;import java.util.Arrays;import java.util.List;/** * Java 9 中新增的管道操作 * @author https://www.wdbyte.com */public class ProcessBuilderTest8 { private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process"; public static void main(String[] args) throws IOException, InterruptedException { ProcessBuilder ls = new ProcessBuilder("/bin/bash", "-c", "ls -l"); ProcessBuilder wc = new ProcessBuilder("wc", "-l"); // 追加日志到文件 File pipeLineLogFile = getFile(BASE_DIR + "/pipe_line_log.txt"); wc.redirectOutput(Redirect.appendTo(pipeLineLogFile)); List processes = ProcessBuilder.startPipeline(Arrays.asList(ls, wc)); Process process = processes.get(processes.size() - 1); System.out.println("pid:" + process.pid()); System.out.println("exitCode:" + process.waitFor()); Files.lines(pipeLineLogFile.toPath()).forEach(System.out::println); } public static File getFile(String filePath) throws IOException { File logFile = new File(filePath); if (!logFile.exists()) { logFile.createNewFile(); } return logFile; }}
这会输出:
pid:33518exitCode:0 21
ProcessBuilder 超时与终止进程有时不能按照自己想要的情况运行,需要对进程进行管理,常见的操作是超时控制以及进程退出。下面通过一个例子来演示如何操作。
先编译一个用于测试的 Java 类 ExecDemo.java
,此类每隔一秒输出一个数字,共输出10个数字,预计需要10s输出完毕。
下面是代码部分:
import java.io.IOException;/** * @author https://www.wdbyte.com */public class ExecDemo { public static void main(String[] args) throws InterruptedException { System.out.println("开始处理数据..."); for (int i = 0; i < 10; i++) { Thread.sleep(1000); System.out.println(i); } System.out.println("数据处理完毕"); }}
再编写一个 ProcessBuilder
来执行 ExceDemo
,但是在执行 3 秒后就判断是否运行完成,如果没有则杀死进程。
package com.wdbyte.os.process;import java.io.File;import java.io.IOException;import java.util.concurrent.TimeUnit;/** * 运行一个 Java 程序 * 等待一定时间后检查状态,未结束则直接杀死进程。 * * @author https://www.wdbyte.com */public class ProcessBuilderTest9 { private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process"; public static void main(String[] args) throws IOException, InterruptedException { ProcessBuilder processBuilder = new ProcessBuilder(); processBuilder.directory(new File(BASE_DIR)); processBuilder.command("java", "ExecDemo.java"); // 把子线程 I/O 输出重定向当前进程 processBuilder.inheritIO(); Process process = processBuilder.start(); // 等待一定时间 boolean waitFor = process.waitFor(3, TimeUnit.SECONDS); System.out.println("waitFor:" + waitFor); // 若未退出,杀死子进程 if (!waitFor) { process.destroyForcibly(); process.waitFor(); System.out.println("杀死进程:" + process); } }}
这会输出:
开始处理数据...01waitFor:false杀死进程:Process[pid=35084, exitValue=137]
在这段代码中,destroyForcibly()
用于杀死进程,但是杀死进程并不是瞬间完成的,所以接着使用 waitFor()
来等待程序真正被杀死退出。
很多情况下,在执行一个命令启动一个新线程后,我们不想阻塞等待进程的完成,想要异步化,在进程执行完成后进行通知回调。这时可以使用 CompletableFuture
来实现这个功能。
package com.wdbyte.os.process;import java.io.File;import java.io.IOException;import java.util.concurrent.CompletableFuture;/** * @author https://www.wdbyte.com */public class ProcessBuilderTest10 { private static String BASE_DIR = "/Users/darcy/git/JavaNotes/core-java-modules/core-java-os/src/main/java/com/wdbyte/os/process"; public static void main(String[] args) throws InterruptedException { ProcessBuilder processBuilder = new ProcessBuilder(); processBuilder.directory(new File(BASE_DIR)); processBuilder.command("java", "ExecDemo.java"); // 把子线程 I/O 输出重定向当前进程 processBuilder.inheritIO(); // 创建 CompletableFuture 对象 CompletableFuture future = CompletableFuture.supplyAsync(() -> { try { // 命令执行 Process process = processBuilder.start(); // 任务超时时间 process.waitFor(); } catch (IOException e) { throw new RuntimeException(e); } catch (InterruptedException e) { throw new RuntimeException(e); } return null; }); // 注册回调函数,处理异步等待的结果 future.thenAccept(result -> { System.out.println("进程执行结束"); }); System.out.println("主进程等待"); Thread.sleep(20 * 1000); }}
这会输出:
主进程等待开始处理数据...0123456789数据处理完毕进程执行结束
ProcessBuilder 总结在这篇文章中,我们详细介绍了 ProcessBuilder 的具体用法,并且给出了常用的操作示例。同时也介绍了 Java 9 开始为 ProcessBuilder 引入的管道操作,最后介绍如何对 Process 进程进行异步处理。
一如既往,文章中代码存放在 Github.com/niumoo/javaNotes.
本文原发于网站:https://www.wdbyte.com/java/os/processbuilder/我的公众号:ProcessBuilder API 使用教程
责任编辑:hnmd003
相关阅读
相关阅读
-
使用 ProcessBuilder API 优化你的流程
Java的ProcessAPI为开发者提供了执行操作系统命令的强大功能,但是某些
-
卖货主播逃离618!怎么联系主播卖货?
卖货主播逃离618!今年的618你购物了吗?对于那些带货主播来说,今年可能并不会太好过。眼下,史上最卷的6...
-
所有权优势是什么意思?所有权优势的主要来源
所有权优势是什么意思?所有权优势,又称垄断优势或厂商优势,是指一国企业拥有或能够得到的而他国企业没...
-
公司债券和企业债券有什么区别?公司债券发行试点办法
公司债券和企业债券有什么区别?1、发行主体不同:公司债券是由股份有限公司或有限责任公司发行的债券;企...
-
雪梨杭州总部大楼将动工;薇娅电商公司申请注销;任泽平减资9000万
雪梨杭州总部大楼将动工;薇娅电商公司申请注销;任泽平减资9000万。
-
华数传媒:将探索尝试虚拟人艾珈在虚拟客服、AI 主播、直播带货等多种场景应用
品玩6月14日讯,华数传媒近日接受机构调研时表示,公司将积极探索尝试
-
Meta 发布“类人”人工智能模型 I-JEPA,能根据对世界的理解填补图像缺失-世界快播报
品玩6月14日讯,Meta公司周二宣布,将向研究人员开放一种新的类人人工
-
机构:一季度全球智能手机产量创 9 年来新低 每日播报
钛媒体App6月14日消息,调研机构集邦咨询(TrendForce)14日发布的一份
-
Meta 发布 Horizon Worlds 114 更新,引入 World Chat 功能
品玩6月14日讯,Meta公司6月14日发布了HorizonWorlds114更新,在修复了
-
世界最新:索尼 Xperia 1 V:“轻薄”与“高画质”兼得的影像旗舰
与传统的相机相比,手机有着轻便小巧的特性,拍摄门槛也更低。然而随着
-
每平 12.6 万,没售楼处却连续 3 次日光,另一边两年“跳水” 900 万…深圳豪宅市场有点魔幻
6月12日9时,深圳网红盘海德园项目正式选房,到当日18时10分左右,不到
-
美国版贴吧 Reddit 被用户“爆吧”,全世界社区都没法活了? 全球讯息
图片来源@视觉中国文|雷科技上一次Reddit出名,可能还是因为……ChatG
-
中国游戏等“春”来 天天快看点
图片来源@视觉中国文|一点财经对于游戏行业来说,2023年将是压力依旧
-
西藏金融机构举办征信知识竞赛
6月12日,由人行拉萨中心支行主办,邮储银行西藏分行承办的“十年征信
-
《西藏自治区产业链链长制工作方案》印发 全力推进产业链向中高端迈进|当前观点
《工作方案》指出,西藏重点产业链包括铜产业链、锂产业链、藏医药产业
-
西藏芒康县纳西民族乡村村有产业、户户有项目、人人有干劲——“擦卡洛”的红火日子-全球报资讯
这几年,纳西民族乡老百姓的生活变化很大。
-
【世界热闻】西藏林芝市工布江达县:“中国藏猪之乡”的致富密码
藏猪又名“藏香猪”“人参猪”,是西藏古老畜种资源的优秀代表。林芝气
-
今年西藏计划实施连片人工种草11.68万亩
近年来,西藏草业科技取得长足发展,有力支撑了草原生态系统保护、草产
-
u盾证书怎么下载?u盾证书下载失败怎么办?
u盾证书怎么下载?在工商银行我的页面点击安全中心。进入到安全中心页面后点击更多。在更多页面点击u盾助...
-
成交回报是什么意思?成交回报和龙虎榜区别
成交回报是什么意思?个股的成交回报中,投资者可以看到买卖单成交的具体情况,成交回报是什么意思?在尤...
-
结汇水单是什么意思?银行回单为啥叫水单?
结汇水单是什么意思?结汇水单是指从国外开立的结汇支票、汇兑支票及有关凭证,是跨境资金流动的重要手段...
-
普通债券基金有哪些?普通债券基金和纯债基金的区别
普通债券基金有哪些?一、标准型、普通型和特定策略型1 标准债券型基金,仅投资于固定预期年化预期收益...
-
H&H国际控股与初步买方就发行5820.7万美元票据订立购买协议 全球报资讯
H&H国际控股(01112)发布公告,于2023年6月13日,公司与初步买方就发
-
抄盘怎么抄?抄盘线是多少日均线?
抄盘怎么抄?股票的根本目标是盈利,而实现的手段则是低买高抛。从最低点买入即为抄盘,不过如何能抄到盘...
-
如何炒黄金白银?为什么炒黄金的人不多?
如何炒黄金白银?第一,要选择正规的交易平台。目前有许多国内公司都从事黄金白银交易,其中有许多非法公...
-
每日热讯!福特电马遭遇惨痛失败,福特正在急剧收缩在华业务
福特电马遭遇惨痛失败,福特正在急剧收缩在华业务种种迹象显示,福特正
-
“谁还在买汉兰达?”
“谁还在买汉兰达?”讲究“传统”的汉兰达真的没有市场了吗?2023年的
-
环球热门:【国际快讯】特斯拉招聘职位数量减少4%;通用和三星SDI将在美国建电池厂;丰田股价创8月来新高
【国际快讯】特斯拉招聘职位数量减少4%;通用和三星SDI将在美国建电池
-
淘宝不能付款是什么原因?淘宝账户支付功能关闭怎么解开?
淘宝不能付款是什么原因?有的买家付款时发现自己的淘宝账号支付不了,就无法继续购买。那么淘宝无法支付...
-
环球快讯:市交通局推进全区通办货运驾驶员资格证业务
日前,经营性道路普通货物运输驾驶员从业人员资格证的核发、补证、换证
精彩推荐
阅读排行
精彩推送
- 热门:昆都仑区:家门口的幸福“...
- 王月飞:守护“声命线” 高效保平安
- 环球速递!财经视点|致富带头人...
- 彩旗招展 喝彩“包马”
- 权证行权是什么意思?退市股不去...
- 20余省已公布高考查分时间!高考...
- 豫光金铅(600531):6月13日北...
- 利润分配怎么计算?利润分配是什...
- 基金历史净值是什么意思?买基金...
- 医保基数是按什么标准缴纳?医保...
- 无责赔付是什么意思?无责赔付人...
- 天津突发爆炸致3死多伤!嫌犯被...
- 浙江慈溪:“千万工程”优胜县的...
- 理想汽车:有信心在 2024 年实...
- AMD 公布新款 MI300X AI 芯片|环球观天下
- MacBook Air 15:情感上拉跨,...
- 特斯拉美国官网:Model Y 价格...
- 谷歌暂缓在欧盟推出聊天机器人 ...
- 车圈撕起来,饭圈都得靠边站
- 股民圈炸了!AI 大牛鸿博股份竟...
- 天天观焦点:理想汽车:有信心在...
- 证券市场属于什么市场?证券市场...
- 留存收益包括什么?留存收益的资...
- 范围经济是什么意思?范围经济与...
- 局长妻子疑吃空饷官方成立调查组...
- 收益性支出是什么意思?收益性支...
- 山推股份06月13日被深股通减持53...
- 碳基金到底是什么意思?碳基金的...
- 监管资本是什么?监管资本和经济...
- 影视公司扎堆搞直播,万元月薪招...