Java代碼執(zhí)行shell命令的實(shí)現(xiàn)
本文描述兩種方式使用java代碼執(zhí)行shell命令,首先使用Runtime類(lèi)調(diào)用exce方法,其次使用ProcessBuilder實(shí)例實(shí)現(xiàn)更靈活的方式。
1. 環(huán)境準(zhǔn)備
執(zhí)行shell命令之前,我們需要獲取jvm底層操作系統(tǒng),同時(shí)定義通用消費(fèi)流的類(lèi)。
1.1. 操作系統(tǒng)依賴(lài)
在創(chuàng)建進(jìn)場(chǎng)執(zhí)行shell命令之前,我們需要獲取jvm運(yùn)行在具體哪個(gè)操作系統(tǒng)之上。因?yàn)閃indows執(zhí)行shell命令是cmd.exe,而其他操作系統(tǒng)發(fā)布標(biāo)準(zhǔn)shell是sh:
boolean isWindows = System.getProperty("os.name").toLowerCase().startsWith("windows");
1.2. 輸入和輸出
此外我們還需要一種方法來(lái)連接進(jìn)程的輸入、輸出流。直到輸出流被消費(fèi)進(jìn)程才會(huì)返回成功,否則會(huì)掛起。下面實(shí)現(xiàn)通用類(lèi)StreamGobbler消費(fèi)InputStream:
private static class StreamGobbler implements Runnable { private InputStream inputStream; private Consumer<String> consumer; public StreamGobbler(InputStream inputStream, Consumer<String> consumer) { this.inputStream = inputStream; this.consumer = consumer; } @Override public void run() { new BufferedReader(new InputStreamReader(inputStream)).lines() .forEach(consumer); } }
該類(lèi)實(shí)現(xiàn)Runnable接口,意味著能夠被任何Executor執(zhí)行。
2. Runtime.exec()執(zhí)行
Runtime.exec()方法是一種生成新子進(jìn)程的簡(jiǎn)單方法,但不能定制。下面示例列出用戶(hù)目錄的目錄清單并打印至控制臺(tái):
String homeDirectory = System.getProperty("user.home"); Process process; if (isWindows) { process = Runtime.getRuntime() .exec(String.format("cmd.exe /c dir %s", homeDirectory)); } else { process = Runtime.getRuntime() .exec(String.format("sh -c ls %s", homeDirectory)); } StreamGobbler streamGobbler = new StreamGobbler(process.getInputStream(), System.out::println); Executors.newSingleThreadExecutor().submit(streamGobbler); int exitCode = process.waitFor(); assert exitCode == 0;
3. ProcessBuilder執(zhí)行
第二種方法使用ProcessBuilder。這比Runtime方法更可取,因?yàn)槟軌蚨ㄖ埔恍┘?xì)節(jié)。
例如:
- 改變正在運(yùn)行Shell命令的工作目錄,使用builder.directory()方法
- 使用builder.environment()方法,設(shè)置自定義鍵值對(duì)作為環(huán)境變量
- 重定向輸入和輸出流值自定義流
- 使用build.inheritio()方法將它們都繼承到當(dāng)前JVM進(jìn)程的流中
ProcessBuilder builder = new ProcessBuilder(); if (isWindows) { builder.command("cmd.exe", "/c", "dir"); } else { builder.command("sh", "-c", "ls"); } builder.directory(new File(System.getProperty("user.home"))); Process process = builder.start(); StreamGobbler streamGobbler = new StreamGobbler(process.getInputStream(), System.out::println); Executors.newSingleThreadExecutor().submit(streamGobbler); int exitCode = process.waitFor(); assert exitCode == 0;
4. 總結(jié)
本文介紹了兩種不同方法執(zhí)行Shell命令。通常如果需要自定義派生流程的執(zhí)行,例如更改其工作目錄,則應(yīng)考慮使用ProcessBuilder。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java8 實(shí)現(xiàn)提取集合對(duì)象的每個(gè)屬性
這篇文章主要介紹了java8 實(shí)現(xiàn)提取集合對(duì)象的每個(gè)屬性方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-02-02Spring?Boot提高開(kāi)發(fā)效率必備工具lombok使用
這篇文章主要為大家介紹了Spring?Boot提高開(kāi)發(fā)效率的必備工具lombok使用方法示例及步驟說(shuō)明,有需要的朋友可以借鑒參考下,希望能夠有所幫助2022-03-03Java 實(shí)戰(zhàn)練手項(xiàng)目之校園超市管理系統(tǒng)的實(shí)現(xiàn)流程
讀萬(wàn)卷書(shū)不如行萬(wàn)里路,只學(xué)書(shū)上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+SSM+Mysql+Maven+Bootstrap實(shí)現(xiàn)一個(gè)校園超市管理系統(tǒng),大家可以在過(guò)程中查缺補(bǔ)漏,提升水平2021-11-11mybatis框架xml下trim中的prefix與suffix等標(biāo)簽的用法
這篇文章主要介紹了mybatis框架xml下trim中的prefix與suffix等標(biāo)簽的用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07使用@Validated 和 BindingResult 遇到的坑及解決
這篇文章主要介紹了使用@Validated 和 BindingResult 遇到的坑及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10