当前位置: 首页 > 编程笔记 >

Java代码执行shell命令的实现

尉迟越
2023-03-14
本文向大家介绍Java代码执行shell命令的实现,包括了Java代码执行shell命令的实现的使用技巧和注意事项,需要的朋友参考一下

本文描述两种方式使用java代码执行shell命令,首先使用Runtime类调用exce方法,其次使用ProcessBuilder实例实现更灵活的方式。

1. 环境准备

执行shell命令之前,我们需要获取jvm底层操作系统,同时定义通用消费流的类。

1.1. 操作系统依赖

在创建进场执行shell命令之前,我们需要获取jvm运行在具体哪个操作系统之上。因为Windows执行shell命令是cmd.exe,而其他操作系统发布标准shell是sh:

boolean isWindows = System.getProperty("os.name").toLowerCase().startsWith("windows");

1.2. 输入和输出

此外我们还需要一种方法来连接进程的输入、输出流。直到输出流被消费进程才会返回成功,否则会挂起。下面实现通用类StreamGobbler消费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);
  }
}

该类实现Runnable接口,意味着能够被任何Executor执行。

2. Runtime.exec()执行

Runtime.exec()方法是一种生成新子进程的简单方法,但不能定制。下面示例列出用户目录的目录清单并打印至控制台:

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执行

第二种方法使用ProcessBuilder。这比Runtime方法更可取,因为能够定制一些细节。

例如:

  • 改变正在运行Shell命令的工作目录,使用builder.directory()方法
  • 使用builder.environment()方法,设置自定义键值对作为环境变量
  • 重定向输入和输出流值自定义流
  • 使用build.inheritio()方法将它们都继承到当前JVM进程的流中
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. 总结

本文介绍了两种不同方法执行Shell命令。通常如果需要自定义派生流程的执行,例如更改其工作目录,则应考虑使用ProcessBuilder。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。

 类似资料:
  • 问题内容: 我正在尝试从GNU / Linux平台上的Java应用程序执行shell命令。问题是,尽管它可以从bash成功运行,但调用另一个Java应用程序的脚本永远不会结束。我试图调试它: 我尝试使用:ProcessBuilder(); 和Runtime.getRuntime()。exec(cmd); 看起来它正在等待完成。有任何想法吗? 谢谢,Laurențiu 问题答案: 您是否正在处理标准

  • 问题内容: 我仍在尝试掌握如何运行linux或Windows shell命令并在node.js中捕获输出的细节。最终,我想做这样的事情… 重要的一点是必须对全局范围的变量(或对象)可用。我尝试了以下功能,但是由于某种原因,我被打印到控制台了。 我很难理解代码在哪里突破了…该模型的非常简单的原型可以工作… 有人可以帮助我了解为什么有效,但是无效吗?FWIW,我需要使用,因为缓冲区限制为200KB。

  • 我有一个案例,我想问我可以解决与Spring壳。我有一个Main.jar应用程序,它有几个部署在Wildly服务器上的Spring任务。在我的例子中,我不能停止或重新部署main.jar,因为必须不停地提供服务。

  • 我没有得到“你好”:请想知道为什么和做什么

  • 问题内容: 我有一个应用程序,该应用程序应该使用一些shell命令将文件从sdcard复制到/ system / media /。它将需要root用户,并且我正在对root用户的设备进行测试。我正在使用运行时来执行Shell命令,但无法正常工作。这是我的运行时所需要的 但是我的日志只显示其中两个没有被拒绝 这两个看起来是sysrw和sysro命令,但是当我触发此代码时,该应用程序仍会请求root权

  • 问题内容: 我可以用来执行诸如“ ”和“ ”之类的shell命令,它们可以正常工作。 但是,当我使用“ ”,“ ”或“ ”时,它不会在标准输出中显示。 日志显示: 这是我的代码: 问题答案: @Adi Tiwari,我找到了原因。 不直接执行Shell命令,而是执行带有参数的可执行文件。“ ”是内置的shell命令。它实际上是带有option 的可执行文件参数的一部分。像这样的命令是实际的可执行文

  • 问题内容: 我正在尝试在我编写并按下的应用程序中从应用程序仿真器终端(您可以在Google Play中找到它)执行此命令enter,因此请编写: 然后再次按,并使用的新功能开始录制屏幕。 因此,我尝试使用以下命令从Java执行相同的代码: 但是不起作用,因为未创建文件。显然,我在装有android kitkat的有根设备上运行。问题出在哪里?我该如何解决?因为从终端仿真器工作并且在Java中不行?

  • 我正在尝试从jenkinsfile执行一组命令。问题是,当我尝试将stdout的值分配给变量时,它不起作用。我尝试了双引号和单引号的不同组合,但到目前为止没有运气。 在这里,我使用最新版本的jenkinsfile以及旧版本执行脚本。将shell命令放入 """ """ 不允许创建新变量,并给出client_name命令不存在的错误。 我想将命令p4客户端-e$client_name的标准输出分配给