java Process 执行批命令 cmd

1、初始化 Process

Process process = new ProcessBuilder(command).redirectErrorStream(true).start();
或者
Process process  = Runtime.getRuntime().exec(cmd);

 redirectErrorStream(true)

错误输出与标准输出合并,以便通过使用Process类的getInputStream()方法可以读取错误并输出

通过子进程所产生的任何错误输出随后由该对象的start()方法启动将与标准输出合并, 这样既可以用Process.getInputStream()方法来读取。此使得更容易与对应的输出相关的错误消息。初始值是false。

2、处理流

int exitcode = process .waitFor();
new PrintStream(process .getErrorStream(), "error").start();
new PrintStream(process .getInputStream(), "input").start();
process创建的子进程没有自己的控制台或终端,其所有的io操作都是通过(输入流、输出流、错误流)重定向到父进程中,如果该可执行程序的输入、输出或者错误输出比较多的话,而由于运行窗口的标准输入、输出等缓冲区有大小的限制,则可能导致子进程阻塞,甚至产生死锁,其解决方法就是在waitfor()方法之前读出窗口的标准输出、输出、错误缓冲区中的内容

所以,建议process .waitFor(); 放在处理流后面等待
new PrintStream(process .getErrorStream(), "error").start();
new PrintStream(process .getInputStream(), "input").start();
int exitcode = process .waitFor();

3、处理多输入交换参数

(1)使用expect4j

 Process process = Runtime.getRuntime().exec("...");
    Expect expect = new ExpectBuilder()
            .withInputs(process.getInputStream())
            .withOutput(process.getOutputStream())
            .withErrorOnTimeout(true)
            .build();
    expect.expect(contains("Password:"));
    expect.sendLine("secret");
    expect.close();

(2)使用Threa.sleep

本来输出流,以为使用(win)\r\n或(linux)\n 模拟回车就可以输入参数。但是不起作用。

Thread.sleep(1000);在输入之间等待一段时间.将脚本在下一个输入之前消耗上个输入

OutputStream outputStream = process.getOutputStream();
outputStream.write(new String("1var").getBytes());
Thread.sleep(1000); //https://www.it1352.com/2342759.html
outputStream.write(new String("1tyh").getBytes());
outputStream.flush();
outputStream.close(); //写入完成后必须得执行 flush() 和 close() 方法,否则进程无法判断写入是否完成,导致写入失败;

(3)使用1024字节数组

一个 SET/P 语句的输入限制为 1024 个字符.我们可以使用它来解决这种情况,通过管道传输一个比 1024 小一个字符的缓冲区.因此,下一个输入的第一个字符将使缓冲区满,这将导致消耗输入并继续执行下一个语句.所以如果你使用这种转换方法,您的代码将按预期工作.事实上,你甚至可以省去刷新缓存

OutputStream outputStream = process.getOutputStream();
outputStream.write(convert("tester"));
outputStream.write(convert("25"));
outputStream.flush();
outputStream.close(); //写入完成后必须得执行 flush() 和 close() 方法,否则进程无法判断写入是否完成,导致写入失败;

public static byte[] convert(String str){
  byte[] asBytes = str.getBytes();
  byte[] result = new byte[1023];
  for(int i = 0; i < asBytes.length; i++)
      result[i] = asBytes[i];
      return result;
  }
}

代码


/**
 * 用于处理Runtime.getRuntime().exec产生的错误流及输出流
 */ 
public class PrintStreamextends Thread { 
    InputStream is; 
    String type; 
    OutputStream os; 

    PrintStream(InputStream is, String type) { 
        this(is, type, null); 
    } 

    PrintStream(InputStream is, String type, OutputStream redirect) { 
        this.is = is; 
        this.type = type; 
        this.os = redirect; 
    } 

    public void run() { 
        InputStreamReader isr = null; 
        BufferedReader br = null; 
        PrintWriter pw = null; 
        try { 
            if (os != null) 
                pw = new PrintWriter(os); 

            isr = new InputStreamReader(is); 
            br = new BufferedReader(isr); 
            String line=null; 
            while ( (line = br.readLine()) != null) { 
                if (pw != null) 
                    pw.println(line); 
                System.out.println(type + ">" + line);     
            } 

            if (pw != null) 
                pw.flush(); 
        } catch (IOException ioe) { 
            ioe.printStackTrace();   
        } finally{ 
            try { 
                pw.close(); 
                br.close(); 
                isr.close(); 
            } catch (IOException e) { 
                e.printStackTrace(); 
            } 
        } 
    } 
}  

参考:

如何使用 Java Process Builder 执行接受多个输入的批处理脚本? - IT屋-程序员软件开发技术分享社区

java - 如何将参数传递给Java Process? - Thinbug 

Java Process详解及实例_罗汉翔的博客-CSDN博客 

java执行bat命令碰到阻塞 java执行bat命令碰到的阻塞问题的解决办法(IT技术)

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值