Runtime.getRuntime.exec线程阻塞问题

一、问题说明:

ava后端执行cmd命令,windows上命令行能够成功,但是java使用exec执行,大部分时间都卡顿,无法执行完,也不报错。

二、阻塞原因

Java执行Runtime.getRuntime().exec()时,JVM会创建一个新的子进程去调用cmd窗口命令,同时该子进程与JVM建立三个连接管道:标准输入流(getInputStream)、标准输出流(getOutputStream)、标准错误流(getErrorStream)。
执行的cmd命令是对rtsp视频流截图,因为并没有参数的输入和得到结果,所以以为没有必要连同管道,虽然cmd窗口能够成功运行,但是运行的过程中会产生警告信息,因此得到的错误流迟迟得不到接受,就进行卡顿。

三、解决方法

主程序中执行命令,对错误流进行记录
 try {
            Process process4 = Runtime.getRuntime().exec(com);
            StreamGobbler errorGobbler = new StreamGobbler(process4.getErrorStream(),"ERROR");
            errorGobbler.start();
            StreamGobbler outGobbler = new StreamGobbler(process4.getInputStream(), "STDOUT");
            outGobbler.start();
            int proc = process4.waitFor();
//            当前线程等待,如果成功返回0,
            if(proc==0){
                System.out.println("成功");
            }else {
                System.out.println("失败");
            }
            if(process4!=null){
                process4.destroy();
            }
        }catch (IOException | InterruptedException e){
            e.printStackTrace();
        }
引入StreamGobbler类
package aiconnect;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
/**
 * 用于处理Runtime.getRuntime().exec产生的错误流及输出流
 *
 */
public class StreamGobbler extends Thread {
    InputStream is;
    String type;
    OutputStream os;

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

    StreamGobbler(InputStream is, String type, OutputStream redirect) {
        this.is = is;
        this.type = type;
        this.os = redirect;
    }
    public void run() {
        InputStreamReader isr = null;
        BufferedReader br = null;
//        pw用来创建一个文件,并向该文件中写入数据
        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{
//            pw.close();
            try {
                br.close();
                isr.close();
            } catch (Exception e) {
                e.printStackTrace();
            }

        }
    }
}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`Runtime.getRuntime().exec()`是Java中用于执行外部进程的方法。它返回一个Process对象,可以使用这个对象控制和监视进程的执行。 语法如下: ```java public Process exec(String command) throws IOException ``` 其中,command是要执行的命令。 例如,执行一个简单的命令: ```java Process process = Runtime.getRuntime().exec("ls"); ``` 这个代码将在当前目录下执行`ls`命令,并返回一个Process对象。 然而,这只是一个简单的例子。实际上,在执行外部进程时,还需要考虑很多其他方面,例如进程的输入、输出、错误处理等。 下面是一个更复杂的例子,演示如何使用`Runtime.getRuntime().exec()`执行一个Python脚本,并读取它的输出: ```java try { Process process = Runtime.getRuntime().exec("python myscript.py"); BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream())); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } } catch (IOException e) { e.printStackTrace(); } ``` 在这个例子中,我们使用`BufferedReader`读取进程的输出,并将其打印到控制台上。如果进程执行时发生错误,我们可以使用`process.getErrorStream()`方法获取错误流,并对其进行处理。 需要注意的是,`Runtime.getRuntime().exec()`方法在执行外部进程时可能会阻塞当前线程,因此在处理输出时需要开启一个新的线程。此外,还需要注意处理输入、输出、错误流的顺序,以避免死锁等问题。 总之,`Runtime.getRuntime().exec()`是一个非常强大的方法,可以在Java中轻松地执行外部命令和脚本。但是,它需要仔细处理各种输入、输出、错误流,以确保进程能够正确执行。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值