调用Runtime.exec方法将产生一个本地的进程

调用Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息. 由于调用Runtime.exec方法所创建的子进程没有自己的终端或控制台,因此该子进程的标准IO(如stdin,stdou,stderr)都通过Process.getOutputStream(),Process.getInputStream(), Process.getErrorStream()方法重定向给它的父进程了.用户需要用这些stream来向 子进程输入数据或获取子进程的输出. 所以正确执行Runtime.exec("ls")的例程如下:
123456789101112
try
{
     process = Runtime.getRuntime().exec (command);
     InputStreamReader ir=newInputStreamReader(process.getInputStream());
     LineNumberReader input = new LineNumberReader (ir);
     String line;
    while ((line = input.readLine ()) != null)
          System.out.println(line);
}
catch (java.io.IOException e){
     System.err.println ("IOException " + e.getMessage());
}




作者:Onlyfor_love
发表时间:2005-4-26 13:19:11

Runtime.exec(“文本“)

文本的内容必须在DOS或平台上可执行,如Runtime.exec(”move c:\\a.txt c:\\b\\”)就肯定执行不了,所以只能用拆衷的办法,把move c:\a.txt c:\b\写在一个批处理文件T内,然后执行:

Runtime.exec(”t.bat:”)
////////////////////////////////////////////////////////////////////////////////////
Java技巧:使用Runtime.exec重定向本地程序调用

作者: BUILDER.COM
Tuesday, December 31 2002 11:58 AM
Java具有使用Runtime.exec对本地程序调用进行重定向的能力,但是用重定向或者管道进行命令调用将会出错。解决这一问题的办法是通过命令shell运行命令。在Java中调用本地程序会破坏平台独立性规则,但是经常需要这么做才行。

以下是一个简单类的范例,展示了在Unix下运行ls命令的情形:
1234567891011121314151617181920212223242526272829
import java.io.BufferedInputStream;
import java.io.IOException;

public class ExecLs {

    static public void main(String[] args) {
         String cmd = "ls"

        try {
             Process ps = Runtime.getRuntime().exec(cmds);
             System.out.print(loadStream(ps.getInputStream()));
             System.err.print(loadStream(ps.getErrorStream()));
        } catch(IOException ioe) {
             ioe.printStackTrace();
        }
    }

    // read an input-stream into a String
    static String loadStream(InputStream in) throws IOException {
        int ptr = 0;
         in = new BufferedInputStream(in);
         StringBuffer buffer = new StringBuffer();
        while( (ptr = in.read()) != -1 ) {
             buffer.append((char)ptr);
        }
        return buffer.toString();
    }

}



上述代码中重要的部分是exec方法和命令字符串ls。本程序将输出运行目录下的列表细节。

那么,如果你想重定向这些细节内容到文件该怎么办?这一命令行的输入应该写成ls > FILE,但是当你将cmd变量改变成这样的话,运行就会出错,如下:

/bin/ls: >: No such file or directory
/bin/ls: FILE: No such file or directory

出错的原因在于额外的参数被直接传送到了ls命令而不是送到实际的命令行。解决这一问题的办法是将cmd串弄成一个字符串数组,并且将你想运行的程序传送到命令shell。

因此,将cmd行改成下面的样子:

String[] cmd = { "sh", "-c", "ls > FILE" };

你将得到一个名为FILE的文件,里面是目录列表。-c参数是告诉它读取随后的字符串,而最后的参数是你要运行的脚本。

在这种情况下,管道也运行良好,所以你可以把命令改成下面的方式:

String[] cmd = { "/bin/sh", "-c", "/bin/ls | grep d > FILE" };

这种形式将给你一个名为FILE的文件,里面是ls条目中包含d的条目。给出sh和ls的全路径有利于提供你的程序的安全性。

虽然使用Runtime.exec不是创建独立于平台的Java的最佳方式,但是有些时候是必要的。使用这种重定向技术有助于走出Runtime.exec的限制。

作者:Onlyfor_love
发表时间:2005-4-26 13:19:42

Runtime运行外部程序的小结- -
使用Runtime.getRuntime().exec()方法可以在java程序里运行外部程序.
该方法有6个可访问版本:
1.exec(String command)
2.exec(String command, String envp[], File dir)
3.exec(String cmd, String envp[])
4.exec(String cmdarray[])
5.exec(String cmdarray[], String envp[])
6.exec(String cmdarray[], String envp[], File dir)

一般的应用程序可以直接使用第一版本,当有环境变量传递的时候使用后面的版本.
其中2和6版本可以传递一个目录,标识当前目录,因为有些程序是使用相对目录的,所以就要使用这个版本.
当要执行批处理的时候,不能直接传递批处理的文件名,而要使用:
cmd.exe /C start 批处理文件名
使用dos命令(比如dir)时也要使用掉调用.

如果想与调用的程序进行交互,那么就要使用该方法的返回对象Process了,通过Process的getInputStream(),getOutputStream(),getErrorStream()方法可以得到输入输出流,然后通过InputStream可以得到程序对控制台的输出信息,通过OutputStream可以给程序输入指令,这样就达到了程序的交换功能.
例子如下:
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869
package com.broadcontact.netadmin.schedule;
import java.io.PrintWriter;
import java.io.PrintStream;
import java.io.IOException;
import java.sql.SQLException;
import java.util.Date;
import java.io.StringReader;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.Reader;
import java.io.File;
import java.io.BufferedWriter;
import java.io.OutputStreamWriter;
/**
* <p>Title: netadmin</p>
* <p>Description: </p>
* <p>Copyright: Copyright (c) 2002</p>
* <p>Company: nm group</p>
* @author Maico(Panghf)
* @version 1.0
*/
public class ExecuteTask implements Runnable {
  private boolean isRunning=true;
  public ExecuteTask() {
  }
  public void run(){
  }
  public static void main(String[] args){
    try {
Process proc=Runtime.getRuntime().exec("cmd.exe");
         BufferedReader read=new BufferedReader(new InputStreamReader(proc.getInputStream()));
      new Thread(new Echo(read)).start();
       PrintWriter out=new PrintWriter(new OutputStreamWriter(proc.getOutputStream()));
       BufferedReader in=new BufferedReader(new InputStreamReader(System.in));
       String instr = in.readLine();
      while(!"exit".equals(instr)){
         instr = in.readLine();
         out.println(instr);
         file://out.println("telnet 192.168.0.1");
         out.flush();
      }
       in.readLine();
       read.close();
       out.close();
    }
    catch (IOException ex) {
       ex.printStackTrace();
    }
  }
}
class Echo implements Runnable {
  private BufferedReader read;
  public Echo(BufferedReader read){
     this.read = read;
  }
  public void run() {
    try {
       String l = read.readLine();
      while (l != null) {
         System.out.println(l);
         l = read.readLine();
      }
       System.out.println("---执行完毕:");
    }
    catch (IOException ex) {
       ex.printStackTrace();
    }
  }
}




作者:Onlyfor_love
发表时间:2005-4-26 13:20:23

如何正确地应用Runtime类调用程序

用Java编写应用时,有时需要在程序中调用另一个现成的可执行程序或系统命令,这时可以通过组合使用Java提供的Runtime类和Process类的方法实现。下面是一种比较典型的程序模式:
...
Process process = Runtime.getRuntime().exec(".\\p.exe");
process.waitfor( );
...
在上面的程序中,第一行的“.\\p.exe”是要执行的程序名,Runtime.getRuntime()返回当前应用程序的Runtime对象,该对象的exec()方法指示Java虚拟机创建一个子进程执行指定的可执行程序,并返回与该子进程对应的Process对象实例。通过Process可以控制该子进程的执行或获取该子进程的信息。第二条语句的目的等待子进程完成再往下执行。
但在windows平台上,如果处理不当,有时并不能得到预期的结果。下面是笔者在实际编程中总结的几种需要注意的情况:
1、执行DOS的内部命令
如果要执行一条DOS内部命令,有两种方法。一种方法是把命令解释器包含在exec()的参数中。例如,执行dir命令,在NT上,可写成exec("cmd.exe /c dir"),在windows 95/98下,可写成“command.exe /c dir”,其中参数“/c”表示命令执行后关闭Dos立即关闭窗口。另一种方法是,把内部命令放在一个批命令my_dir.bat文件中,在Java程序中写成exec("my_dir.bat")。如果仅仅写成exec("dir"),Java虚拟机则会报运行时错误。前一种方法要保证程序的可移植性,需要在程序中读取运行的操作系统平台,以调用不同的命令解释器。后一种方法则不需要做更多的处理。
2、打开一个不可执行的文件
打开一个不可执行的文件,但该文件存在关联的应用程序,则可以有两种方式。 以打开一个word文档a.doc文件为例,Java中可以有以下两种写法:
exec("start .\\a.doc");
exec(" c:\\Program Files\\Microsoft Office\\office\\winword.exe .\\a.doc");
显然,前一种方法更为简捷方便。
3、执行一个有标准输出的DOS可执行程序
在windows平台上,运行被调用程序的DOS窗口在程序执行完毕后往往并不会自动关闭,从而导致Java应用程序阻塞在waitfor( )。导致该现象的一个可能的原因是,该可执行程序的标准输出比较多,而运行窗口的标准输出缓冲区不够大。解决的办法是,利用Java提供的Process 类提供的方法让Java虚拟机截获被调用程序的DOS运行窗口的标准输出,在waitfor()命令之前读出窗口的标准输出缓冲区中的内容。一段典型的程序如下:
...
String ls_1;
Process process = Runtime.getRuntime().exec("cmd /c dir \\windows");
BufferedReader bufferedReader = new BufferedReader( \
new InputStreamReader(process.getInputStream());
while ( (ls_1=bufferedReader.readLine()) != null)
System.out.println(ls_1);
 
process.waitfor( );

你可能感兴趣的:(java,C++,c,dos,C#)