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。

  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Java执行shell命令可以使用Runtime类或ProcessBuilder类。以下是使用Runtime执行shell命令的示例代码: ```java import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; public class ShellCommand { public static void main(String[] args) { try { // 执行ls命令 Process process = Runtime.getRuntime().exec("ls"); // 读取命令输出结果 BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream())); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } // 等待命令执行完成 int exitCode = process.waitFor(); System.out.println("命令执行完成,退出码:" + exitCode); } catch (IOException | InterruptedException e) { e.printStackTrace(); } } } ``` 使用ProcessBuilder执行shell命令的示例代码如下: ```java import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.util.ArrayList; import java.util.List; public class ShellCommand { public static void main(String[] args) { try { // 构建命令 List<String> command = new ArrayList<>(); command.add("ls"); command.add("-l"); // 执行命令 ProcessBuilder processBuilder = new ProcessBuilder(command); Process process = processBuilder.start(); // 读取命令输出结果 BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream())); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } // 等待命令执行完成 int exitCode = process.waitFor(); System.out.println("命令执行完成,退出码:" + exitCode); } catch (IOException | InterruptedException e) { e.printStackTrace(); } } } ``` 以上代码都是在Linux或MacOS上执行的示例,若在Windows上需要将命令改为相应的Windows命令

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值