纯净、安全、绿色的下载网站

首页|软件分类|下载排行|最新软件|IT学院

当前位置:首页IT学院IT技术

java线程阻塞问题 java调用process线程阻塞问题的解决

遗失的岁月   2021-06-09 我要评论
想了解java调用process线程阻塞问题的解决的相关内容吗遗失的岁月在本文为您仔细讲解java线程阻塞问题的相关知识和一些Code实例欢迎阅读和指正我们先划重点:java阻塞,调用process,java线程阻塞下面大家一起来学习吧

java调用process线程阻塞问题

项目需求中涉及java调用.bat文件进行图像处理先直接上简略版程序

public void draw(){
        //调用bat脚本进行图像处理
        Process process = null;
        InputStream in = null;
        try {
            process = Runtime.getRuntime().exec("startup.bat");
            //输出测试
//            in = process.getInputStream();
//            String line;
//            BufferedReader br = new BufferedReader(new InputStreamReader(in));
//            while ((line = br.readLine()) != null) {
//                System.out.println(line);
//            }
            //等待
            process.waitFor();
        } catch (Exception e) {
        } finally {
            process.destroy();
        }
    }

JAVA使用遇到的问题描述

一般需要调用系统命令时大部分人第一反应肯定是使用Runtime.getRuntime().exec(command)返回一个process对象再调用process.waitFor()来等待命令执行结束获取执行结果

调试的时候发现异常现象process.waitFor();一直没有结束导致线程阻塞再次强行关闭程序后发现图像处理只进行了一部分

根据现象并查看了JDK的帮助文档如下

如有必要一直要等到由该 Process 对象表示的进程已经终止如果已终止该子进程此方法立即返回但是直接调用这个方法会导致当前线程阻塞直到退出子进程

对此JDK文档上还有如此解释:因为本地的系统对标准输入和输出所提供的缓冲池有效所以错误的对标准输出快速的写入何从标准输入快速的读入都有可能造成子进程的阻塞甚至死锁

Process执行逻辑

* 主进程中调用Runtime.exec会创建一个子进程用于执行脚本子进程创建后会和主进程分别独立运行

* 创建的子进程没有自己的终端或控制台它的所有标准 io(即 stdin、stdout 和 stderr)操作都将通过三个流 (getOutputStream()、getInputStream() 和 getErrorStream()) 重定向到父进程父进程使用这些流来提供到子进程的输入和获得从子进程的输出

* 这时候子进程不断向主进程发生数据而主进程调用Process.waitfor后已挂起当前子进程和主进程之间的缓冲区塞满后子进程不能继续写数据然后也会挂起

* 这样子进程等待主进程读取数据主进程等待子进程结束两个进程相互等待最终导致死锁

解决方法:

在waitFor()之前利用单独两个线程分别处理process的getInputStream()和getErrorSteam()防止缓冲区被撑满导致阻塞

修改后代码

public class test {
    public void draw(){
        //调用bat脚本进行图像处理
        Process process = null;
        InputStream in = null;
        try {
            process = Runtime.getRuntime().exec("startup.bat");
            //输出测试
//            in = process.getInputStream();
//            String line;
//            BufferedReader br = new BufferedReader(new InputStreamReader(in));
//            while ((line = br.readLine()) != null) {
//                System.out.println(line);
//            }
            //新启两个线程
            new DealProcessSream(process.getInputStream()).start();
            new DealProcessSream(process.getErrorStream()).start();
            process.waitFor();
        } catch (Exception e) {
        } finally {
            process.destroy();
        }
    }
}
public class DealProcessSream extends Thread {
    private InputStream inputStream;
    public DealProcessSream(InputStream inputStream) {
        this.inputStream = inputStream;
    }
    public void run() {
        InputStreamReader inputStreamReader = null;
        BufferedReader br = null;
        try {
            inputStreamReader = new InputStreamReader(
                    inputStream);
            br = new BufferedReader(inputStreamReader);
            // 打印信息
//            String line = null;
//            while ((line = br.readLine()) != null) {
//                System.out.println(line);
//            }
            // 不打印信息
           while (br.readLine() != null);
        } catch (IOException ioe) {
            ioe.printStackTrace();
        }finally {
            try {
                br.close();
                inputStreamReader.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

Process对象.waitFor()的阻塞问题(坑)

有时需要在程序中调用可执行程序或脚本命令:

Process process = Runtime.getRuntime().exec(shPath);
int exitCode = process .waitFor();

Runtime.getRuntime()返回当前应用程序的Runtime对象该对象的exec()方法指示Java虚拟机创建一个子进程执行指定的可执行程序

并返回与该子进程对应的Process对象实例通过Process可以控制该子进程的执行或获取该子进程的信息

它的所有标准io(即stdinstdoutstderr)操作都将通过三个流(getOutputStream()getInputStream()getErrorStream())重定向到父进程

父进程使用这些流来提供到子进程的输入和获得从子进程的输出因为有些本机平台仅针对标准输入和输出流提供有限的缓冲区大小如果读

写子进程的输出流或输入流出现失败则可能导致子进程阻塞甚至产生死锁(如果程序不断在向标准输出流和标准错误流写数据而JVM不读取的话当缓冲区满之后将无法继续写入数据最终造成阻塞在waifor()这里)

process .getErrorStream():获得子进程的错误输出流

process .getInputStream():获得子进程的普通输出流

简单示例:

Process shellProcess = null;
    try {
 
      shellProcess = Runtime.getRuntime().exec(shPath);
      shellErrorResultReader = new BufferedReader(new InputStreamReader(shellProcess.getErrorStream()));
      shellInfoResultReader =  new BufferedReader(new InputStreamReader(shellProcess.getInputStream()));
      String infoLine;
      while ((infoLine = shellInfoResultReader.readLine()) != null) {
        logger.info("脚本文件执行信息:{}", infoLine);
      }
      String errorLine;
      while ((errorLine = shellErrorResultReader.readLine()) != null) {
        logger.warn("脚本文件执行信息:{}", errorLine);
      }
      // 等待程序执行结束并输出状态
      exitCode = shellProcess.waitFor();
      if (0 == exitCode) {
        logger.info("脚本文件执行成功:" + exitCode);
      } else {
        logger.error("脚本文件执行失败:" + exitCode);
      }
    } catch (Exception e) {
      logger.error("shell脚本执行错误", e);
    } finally {
      if (null != shellInfoResultReader) {
        try {
          shellInfoResultReader.close();
        } catch (IOException e) {
          logger.error("流文件关闭异常:", e);
        }
      }
      if (null != shellErrorResultReader) {
        try {
          shellErrorResultReader.close();
        } catch (IOException e) {
          logger.error("流文件关闭异常:", e);
        }
      }
      if (null != shellProcess) {
        shellProcess.destroy();
      }
}

以上为个人经验希望能给大家一个参考也希望大家多多支持


相关文章

猜您喜欢

  • spring方法级参数校验 浅谈spring方法级参数校验(@Validated)

    想了解浅谈spring方法级参数校验(@Validated)的相关内容吗Leonarde_WP在本文为您仔细讲解spring方法级参数校验的相关知识和一些Code实例欢迎阅读和指正我们先划重点:spring方法,参数校验,@Validated下面大家一起来学习吧..
  • C++类内存分布 解析C++类内存分布

    想了解解析C++类内存分布的相关内容吗lsgxeva在本文为您仔细讲解C++类内存分布的相关知识和一些Code实例欢迎阅读和指正我们先划重点:c++,内存下面大家一起来学习吧..

网友评论

Copyright 2020 www.fresh-weather.com 【世纪下载站】 版权所有 软件发布

声明:所有软件和文章来自软件开发商或者作者 如有异议 请与本站联系 点此查看联系方式