Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 705|回复: 0

Java调用外部程序命令

[复制链接]

该用户从未签到

发表于 2011-9-12 22:37:11 | 显示全部楼层 |阅读模式
今天要写个远程重启服务的功能,为了开发速度,暂时定为java代码+WMIC命令的方法,简单的说,就是利用Java调用本机应用程序的方法。涉及到的 Java类有java.lang包里面的Runtime、Process、ProcessBuilder三个类,以及wmic中重启服务的命令。因为之前 也写过这方面的东西,所以很习惯性的写出了代码:
  Process p = Runtime.getRuntime().exec("wmic ...");
  BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream()));
  String tmp = null;
  while ((tmp = br.readline()) != null) {
  System.out.println(tmp);
  }
  int exitValue = p.waitfor();
  Process p = Runtime.getRuntime().exec("wmic ..."); BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream())); String tmp = null; while ((tmp = br.readline()) != null) { System.out.println(tmp); } int exitValue = p.waitfor();
  运行,结果发现程序不能退出,Debug发现程序阻塞在br.readline()中了,强制结束程序,发现重启服务的命令正常下下去了,去掉程序中获得标准输出的地方和获得返回结果的地方,命令也能正常下去,而且正常退出。
  为什么程序会阻塞呢?Google了一下,发现了大家的解释,应该也是比较权威的解释吧:每个进程都有自己的标准输入、标准输出、标准错误输出,对于某些 依赖于OS的进程,可能其输出缓冲区很小,如果不能及时的读出(标准输出、标注错误输出),将导致进程不能正常退出。我的程序中标准输出已经读了,显然原 因不是这个,难道是错误输出缓冲区中的数据没有读出导致的?带着这个疑问,对程序作了一些更改:
  ProcessBuilder pb = new ProcessBuilder("wmic",...);
  pb.redirectErrorStream(true);
  Process p = pb.start();
  BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream()));
  String tmp = null;
  while ((tmp = br.readline()) != null) {
  System.out.println(tmp);
  }
  int exitValue = p.waitfor();
  ProcessBuilder pb = new ProcessBuilder("wmic",...); pb.redirectErrorStream(true); Process p = pb.start(); BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream())); String tmp = null; while ((tmp = br.readline()) != null) { System.out.println(tmp); } int exitValue = p.waitfor();
  编译运行,发现还是有问题,依然还是阻塞。又google了一下,大家的评论大多还是关于标准输出和标准错误输出,那这程序应该是没有问题了。后来在 cmd中敲入wmic的命令,发现wmic命令敲入以后会进入一个自有的提示符中,难道是因为标准输入的问题。后来又google了一下,验证了我的猜 想,果然是因为wmic进程会等待标准输入,而程序中没有处理标准输入的地方,是标准输入阻塞了进程的退出,修改代码:
  ProcessBuilder pb = new ProcessBuilder("wmic",...);
  pb.redirectErrorStream(true);
  Process p = pb.start();
  p.getOutputStream().close();
  BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream()));
  String tmp = null;
  while ((tmp = br.readline()) != null) {
  System.out.println(tmp);
  }
  int exitValue = p.waitfor();
  ProcessBuilder pb = new ProcessBuilder("wmic",...); pb.redirectErrorStream(true); Process p = pb.start(); p.getOutputStream().close(); BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream())); String tmp = null; while ((tmp = br.readline()) != null) { System.out.println(tmp); } int exitValue = p.waitfor();
  编译运行,程序成功执行。果然是标准输入的原因。
  后来执行的过程中换了一个服务的名称,发现执行失败(能够正常退出,但是返回的结果是“无效动作”),但是同样的命令,在命令行中执行成功,而且直接适用 Runtime.exec()方法中写入整个命令也能够执行成功,难道是ProcessBuilder的错误,ProcessBuilder构造函数有两 个:
  ProcessBuilder(List command)
  利用指定的操作系统程序和参数构造一个进程生成器。
  ProcessBuilder(String... command)
  利用指定的操作系统程序和参数构造一个进程生成器。
  找到ProcessBuilder的源代码,发现了对List的解析方法:JDK将List中的所有字符串用空格连接,对 list中的每个字符串JDK先判断串中是否包含空格,如果包含空格,用双引号将该字符串引起来,再拼到前面字符串的后面(应该是为了解决路径中包含空格 的问题),可恰好Wmic命令的参数中有一段是name="ServiceName",如果ServiceName中包换空格,JDK就会把 name="service name"的外层加一个双引号,导致wmic不能解析该命令了。
  问题终于全都解决了,耗费了多半天的时间,不过收获总是有的,这里总结一下,在使用Java调用外部命令的时候,一定要注意对标准输出、标准输入和错误输 出的处理。对于一般的命令,只需要将标准输出和错误输出合并,一起读出来或者在另外的线程中读出来,而对于一些特殊的命令,还有处理标准输入。建议即使不 使用标准输入,先close了,总是不会出错了。另外,使用ProcessBuilder时要注意它的空格处理方式是否是你想要的,如果不是,就不能用 ProcessBuilder了,直接使用Runtime.exec()就好了。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

GMT+8, 2025-1-16 02:50 , Processed in 0.403591 second(s), 36 queries .

Powered by Discuz! X3.4

© 2001-2017 Comsenz Inc.

快速回复 返回顶部 返回列表