在模拟cmd调用Python时遇到一些情况,这类问题可以归类为“超时,阻塞”等,问题原因:

Process p=Runtime.getRuntime().exec(String[] cmd);

Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息。 由于调用Runtime.exec方法所创建的子进程没有自己的终端或控制台,因此该子进程的标准IO(如stdin,stdou,stderr)都通过p.getOutputStream(),p.getInputStream(),p.getErrorStream() 方法重定向给它的父进程了.用户需要用这些stream来向子进程输入数据或获取子进程的输出。

例如:Runtime.getRuntime().exec("ls") 另外需要关心的是Runtime.getRuntime().exec()中产生停滞(阻塞,blocking)的问题? 这个是因为Runtime.getRuntime().exec()要自己去处理stdout和stderr的输出, 就是说,执行的结果不知道是现有错误输出(stderr),还是现有标准输出(stdout)。你无法判断到底那个先输出,所以可能无法读取输出,而一直阻塞。 例如:你先处理标准输出(stdout),但是处理的结果是先有错误输出(stderr), 一直在等错误输出(stderr)被取走了,才到标准输出(stdout),这样就产生了阻塞。

解决办法:

用两个线程将标准输出(stdout)和错误输出(stderr)。

完整代码:

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.InputStreamReader;

/**
 * Create by yster@foxmail.com 2018/11/9 0009 22:28
 */
public class ExecuteCmd {
    /**
     * 执行外部程序,并获取标准输出
     */
    public static String execute(String[] cmd,String... encoding) {
        BufferedReader bufferedReader;
        InputStreamReader inputStreamReader;
        try {
            Process p = Runtime.getRuntime().exec(cmd);

            /* 为"错误输出流"单独开一个线程读取之,否则会造成标准输出流的阻塞 */
            Thread t = new Thread(new InputStreamRunnable(p.getErrorStream(), "ErrorStream"));
            t.start();

            /* "标准输出流"就在当前方法中读取 */
            BufferedInputStream bis = new BufferedInputStream(p.getInputStream());

            if (encoding != null && encoding.length != 0) {
                inputStreamReader = new InputStreamReader(bis, encoding[0]);// 设置编码方式
            } else {
                inputStreamReader = new InputStreamReader(bis, "utf-8");
            }
            bufferedReader = new BufferedReader(inputStreamReader);

            StringBuilder sb = new StringBuilder();
            String line;

            while ((line = bufferedReader.readLine()) != null) {
                sb.append(line);
                sb.append("\n");
            }

            bufferedReader.close();
            p.destroy();
            return sb.toString();
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }

}
import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;

class InputStreamRunnable implements Runnable {
    private BufferedReader bReader = null;

    InputStreamRunnable(InputStream is, String type) {
        try {
            bReader = new BufferedReader(new InputStreamReader(new BufferedInputStream(is), "UTF-8"));
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }

    public void run() {
        String line;
        int num = 1;
        try {
            while ((line = bReader.readLine()) != null) {
                //System.out.println("---->"+String.format("%02d",num++)+" "+line);
            }
            bReader.close();
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
}

最新文章

  1. 第6届蓝桥杯javaA组第7题,牌型种数,一道简单的题带来的思考
  2. 36 网络相关函数(四)——live555源码阅读(四)网络
  3. C# 字符流打印类
  4. sshuttle基于VPN的透明代理,安全连接
  5. centos6的安装
  6. 进程cookie与硬盘cookie
  7. python-集合内置函数详解
  8. JVM-类加载器
  9. 啰嗦的 java,简洁的 lombok —— lombok 的使用及简单实现单例模式注解
  10. [HTML]HTML隐藏文本框的四种方式
  11. 【perl】企业微信发消息
  12. 设置Vmware中Kali_linux 共享文件夹
  13. 如何猜出 Y combinator
  14. operator new,new operator,placement new的区别
  15. 数据库类型空间效率探索(三)-char
  16. 【转】Centos yum 换源
  17. [Python_2] Python 基础
  18. 【FastDFS】FastDFS在CentOS的搭建
  19. tyvj1391走廊泼水节
  20. 打造 Laravel 优美架构 谈可维护性与弹性设计

热门文章

  1. 高强度学习训练第二天总结:Opencv+Android+CameraView小demo
  2. E203 同步fifo
  3. 读oc52个有效方法的总结
  4. loadView的原理
  5. Android Activity之间的数据传递
  6. Mac下安装npm全局包提示权限不够
  7. Context都没弄明白,还怎么做Android开发
  8. DDL创建数据库,表以及约束(极客时间学习笔记)
  9. 编译器与Debug的传奇:Grace Murray Hopper小传
  10. GCN