@ProcessBuilder 、Runtime和Process 的区别
1、版本原因
ProcessBuilder是从java1.5加进来的,而exec系列方法是从1.0开始就有的,后续版本不断的重载这个方法,到了1.5已经有6个之多。
2、ProcessBuilder、Runtime和Process 三者之间的联系
ProcessBuilder.start() 和 Runtime.exec() 方法都被用来创建一个操作系统进程(执行命令行操作),并返回 Process 子类的一个实例,该实例可用来控制进程状态并
获得相关信息。
3、ProcessBuilder和Process区别
既然有Process类,那为什么还要发明个ProcessBuilder类呢?ProcessBuilder和Process两个类有什么区别呢?
首先,每个进程生成器(即ProcessBuilder对象)管理这些进程属性:
命令:是一个字符串列表,它表示要调用的外部程序文件及其参数(如果有)。在此,表示有效的操作系统命令的字符串列表是依赖于系统的。例如,每一个总体变
量,通常都要成为此列表中的元素,但有一些操作系统,希望程序能自己标记命令行字符串——在这种系统中,Java 实现可能需要命令确切地包含这两个元素。
环境:是从变量 到值 的依赖于系统的映射。初始值是当前进程环境的一个副本(请参阅 System.getenv())。
工作目录:默认值是当前进程的当前工作目录,通常根据系统属性 user.dir 来命名。
redirectErrorStream 属性:最初,此属性为 false,意思是子进程的标准输出和错误输出被发送给两个独立的流,这些流可以通过 Process.getInputStream() 和
Process.getErrorStream() 方法来访问。如果将值设置为 true,标准错误将与标准输出合并。这使得关联错误消息和相应的输出变得更容易。在此情况下,合并的数据可从 Process.getInputStream() 返回的流读取,而从 Process.getErrorStream() 返回的流读取将直接到达文件尾。
所以就有,ProcessBuilder为进程提供了更多的控制,例如,可以设置当前工作目录,还可以改变环境参数。而Process的功能相对来说简单的多。
ProcessBuilder是一个final类,有两个带参数的构造方法,你可以通过构造方法来直接创建ProcessBuilder的对象。而Process是一个抽象类,一般都通过Runtime.ex
ec()和ProcessBuilder.start()来间接创建其实例。
注意:
修改进程构建器的属性将影响后续由该对象的 start() 方法启动的进程,但从不会影响以前启动的进程或 Java 自身的进程。
ProcessBuilder类不是同步的。如果多个线程同时访问一个 ProcessBuilder,而其中至少一个线程从结构上修改了其中一个属性,它必须 保持外部同步。
4、ProcessBuilder 、Runtime区别
ProcessBuilder.start() 和 Runtime.exec()传递的参数有所不同,Runtime.exec()可接受一个单独的字符串,这个字符串是通过空格来分隔可执行命令程序和参数的;
也可以接受字符串数组参数。而ProcessBuilder的构造函数是一个字符串列表或者数组。列表中第一个参数是可执行命令程序,其他的是命令行执行是需要的参数。
通过查看JDK源码可知,Runtime.exec最终是通过调用ProcessBuilder来真正执行操作的。
- import java.io.BufferedReader;
- import java.io.File;
- import java.io.IOException;
- import java.io.InputStream;
- import java.io.InputStreamReader;
- import java.util.Map;
- public class ProcessBuilderTest {
- public static void restart() throws IOException {
- // Runtime 例子
- Process p;
- // test.bat中的命令是ipconfig/all
- String cmd = "c:\\test\\test.bat";
- try {
- // 执行命令
- p = Runtime.getRuntime().exec(cmd);
- // 取得命令结果的输出流
- InputStream fis = p.getInputStream();
- // 用一个读输出流类去读
- InputStreamReader isr = new InputStreamReader(fis);
- // 用缓冲器读行
- BufferedReader br = new BufferedReader(isr);
- String line = null;
- // 直到读完为止
- while ((line = br.readLine()) != null) {
- System.out.println(line);
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- // ProcessBuilder 例子 Java程序自重启
- // 用一条指定的命令去构造一个进程生成器
- ProcessBuilder pb = new ProcessBuilder("java", "-jar", "Test3.jar");
- // 让这个进程的工作区空间改为F:\dist
- // 这样的话,它就会去F:\dist目录下找Test.jar这个文件
- pb.directory(new File("F:\\dist"));
- // 得到进程生成器的环境 变量,这个变量我们可以改,
- // 改了以后也会反应到新起的进程里面去
- Map<String, String> map = pb.environment();
- Process p1 = pb.start();
- // 然后就可以对p做自己想做的事情了
- // 自己这个时候就可以退出了
- System.exit(0);
- }
- }
- class PBDemo {
- public static void main(String args[]) {
- try {
- ProcessBuilder proc = new ProcessBuilder("notepad.exe", "testfile");
- proc.start();
- } catch (Exception e) {
- System.out.println("Error executing notepad.");
- }
- }
- }