javashell脚本怎么写( 二 )


原因是:可执行程序的输出可能会比较多,而运行窗口的输出缓冲区有限,会造成waitFor一直阻塞 。解决的办法是,利用Java提供的Process类提供的getInputStream,getErrorStream方法让Java虚拟机截获被调用程序的标准输出、错误输出,在waitfor()命令之前读掉输出缓冲区中的内容 。
6.想写一个shell脚本,内容是一个创建目录的,还有java如何调用的,求public static synchronized void www() throws Exception {
Process pro = null;
String cmd = "D:/database/sqlpad/sqlpad.cmd";
pro = Runtime.getRuntime().exec(cmd);
new DoOutput(pro.getInputStream()).start();
new DoOutput(pro.getErrorStream()).start();
}
private static class DoOutput extends Thread {
public InputStream is;
public DoOutput(InputStream is) {
this.is = is;
}
public void run() {
BufferedReader br = new BufferedReader(new InputStreamReader(this.is));
String str = null;
StringBuilder sb = new StringBuilder();
try {
while ((str = br.readLine()) != null){
sb.append(str+"\n");
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if (br != null) {
try {
br.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}
7.怎么通过java去调用并执行shell脚本以及问题总结对于第一个问题:java抓取,并且把结果打包 。
那么比较直接的做法就是,java接收各种消息(db,metaq等等),然后借助于jstorm集群进行调度和抓取 。最后把抓取的结果保存到一个文件中,并且通过调用shell打包, 回传 。
也许有同学会问, 为什么不直接把java调用odps直接保存文件,答案是,我们的集群不是hz集群,直接上传odps速度很有问题,因此先打包比较合适 。(这里不纠结设计了,我们回到正题)java调用shell的方法通过ProcessBuilder进行调度这种方法比较直观,而且参数的设置也比较方便, 比如我在实践中的代码(我隐藏了部分业务代码):ProcessBuilderpb = new ProcessBuilder("./" + RUNNING_SHELL_FILE, param1,param2, param3);pb.directory(new File(SHELL_FILE_DIR));int runningStatus = 0;String s = null;try {Process p = pb.start();try {runningStatus = p.waitFor();} catch (InterruptedException e) {}} catch (IOException e) {}if (runningStatus != 0) {}return;这里有必要解释一下几个参数:RUNNING_SHELL_FILE:要运行的脚本SHELL_FILE_DIR:要运行的脚本所在的目录; 当然你也可以把要运行的脚本写成全路径 。
runningStatus:运行状态,0标识正常 。详细可以看java文档 。
param1, param2, param3:可以在RUNNING_SHELL_FILE脚本中直接通过1,2,$3分别拿到的参数 。直接通过系统Runtime执行shell这个方法比较暴力,也比较常用, 代码如下:p = Runtime.getRuntime().exec(SHELL_FILE_DIR + RUNNING_SHELL_FILE + " "+param1+" "+param2+" "+param3);p.waitFor();我们发现,通过Runtime的方式并没有builder那么方便,特别是参数方面,必须自己加空格分开,因为exec会把整个字符串作为shell运行 。
可能存在的问题以及解决方法如果你觉得通过上面就能满足你的需求,那么可能是要碰壁了 。你会遇到以下情况 。
没权限运行这个情况我们团队的朱东方就遇到了, 在做DTS迁移的过程中,要执行包里面的shell脚本, 解压出来了之后,发现执行不了 。那么就按照上面的方法授权吧java进行一直等待shell返回这个问题估计更加经常遇到 。
原因是, shell脚本中有echo或者print输出, 导致缓冲区被用完了! 为了避免这种情况, 一定要把缓冲区读一下, 好处就是,可以对shell的具体运行状态进行log出来 。比如上面我的例子中我会变成:ProcessBuilderpb = new ProcessBuilder("./" + RUNNING_SHELL_FILE, keyword.trim(),taskId.toString(), fileName);pb.directory(new File(CASPERJS_FILE_DIR));int runningStatus = 0;String s = null;try {Process p = pb.start(); = new BufferedReader(new InputStreamReader(p.getInputStream())); = new BufferedReader(new InputStreamReader(p.getErrorStream()));while ((s = stdInput.readLine()) != null) {LOG.error(s);}while ((s = stdError.readLine()) != null) {LOG.error(s);}try {runningStatus = p.waitFor();} catch (InterruptedException e) {}记得在start()之后, waitFor()之前把缓冲区读出来打log, 就可以看到你的shell为什么会没有按照预期运行 。