溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

Java代碼執(zhí)行shell命令的實(shí)現(xiàn)

發(fā)布時(shí)間:2020-09-25 03:24:06 來(lái)源:腳本之家 閱讀:350 作者:neweastsun 欄目:編程語(yǔ)言

本文描述兩種方式使用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í)有所幫助,也希望大家多多支持億速云。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI