JAVA 【精】从底层分析RunTime出错,Cmd命令执行正常

奥特曼超人杜锦阳曾经说过:“宁可在法度外灭亡,不在法度中生存。”

深圳市米奇云科技有限公司



Cmd命令执行失败,可能大家开发中经常会有遇到如下问题,可是百度谷歌却出不来,博主踩的坑共享给大家:

  • 报错 CreateProcess error=2, ϵͳÕҲ»µ½ָ¶
  • JAVA cmd执行失败 JAVA
  • RunTime报错,cmd下却执行正常
  • ProcessBuilder异常CreateProcess error=2, ϵͳÕҲ»µ½ָ¶
  • JAVA CMD.EXE /C 的问题


     这里使用的是数组命令,这里记录下这些问题,最近遇到一个比较变态问题,这里是要做个 在线安卓多渠道打包 的东西,中间涉及解包,回编等等……
     主要遇到的问题是传入的参数,怎么空格都不行,折腾了一上午,后来看了底层代码才搞定,先来总结下解决的过程和方法。

    1. 假设你要执行 cmd.exe /C 的命令,这里记住,如果执行的是外部 .exe .bat 之类的,一定不要在数组面前加 CMD /C,如果加了,那就是 执行命令失败 !

    2. 假设你要执行多个参数的,记住,不要学网上的博客在数组命令前加“”,这个会挂逼。

    3. 假设你要执行多个参数的,不用和PC上Cmd一样去空格,之前我也是尝试了,发现不行去看的底层源码才发现,底层会读取空格并换成“”,底下会把源码贴出来。

    4. 最后一点,只要当成参数传入替换即可,不要主动去空格!


一、 CreateProcess error=2, ϵͳÕҲ»µ½ָ¶

 这个报错有2种常见方式,常见的是直接用 CMD.EXE\C 之后调用了外部的exe

//执行代码
Runtime.getRuntime().exec(cmds);
  • 第一种: 这里传入的错误参数:”D:/dujinyang/immqy.exe
    d”,”k”,”D:/dujinyang/immqy_new”,为什么会出现这种,因为指令之间如果有空格而不用不同的字符串隔开,就会无法识别指令。

  • 第二种: 这里传入的参数都没有问题,只是位符错误,所以才报了这种错,这个时候就要换成其它命令来尝试这种错误,而且,CMD/C
    接收的是默认的参数,如果带exe,会被识别成一个程序,如果带参数,除非是bat能直接接收,否则只会当成一个程序处理,这里如果用apktool也会出现这种情况。


二、cmd执行失败

  • 第一种:参数错误,不用多说,解决办法,在string[]组里替换掉自己的参数来测试,如果测试OK再还原占位符。
  • 第二种:占位符错误,解决办法,还是替换参数来测试。

    例子:

//正确代码
String[] CMD_APKTOOL_D = new String [] { "CMD.EXE", "/C","%1","d","-f","%2","-o","%3"};//解压


/**以下都是错误代码方式**/

String[] CMD_APKTOOL_D = new String [] { "CMD.EXE", "/C"," %1"," d"," -f"," %2"," -o","%3"};//解压

String[] CMD_APKTOOL_D = new String [] { "CMD.EXE", "/C","%1"," d "," -f ","%2"," -o","%3"};//解压

String[] CMD_APKTOOL_D = new String [] { "CMD.EXE", "/C","%1","","d","","-f","","%2","","-o","","%3"};//解压


三、源码分析

 为什么会出现这么多的错误,而且空格符又有这么多问题,查了ProcessBuilder相关的API,JAVA RUNTIME什么都看了,没发现什么问题,后来调试中在ProcessImpl.class发现了一段代码,看下图

/*
 * @(#)ProcessImpl.java 1.32 06/03/22
 *
 * Copyright 2006 Sun Microsystems, Inc. All rights reserved.
 * SUN PROPRIETARY/CONFIDENTIAL. Use is subject to license terms.
 */

package java.lang;

import java.io.*;

/* This class is for the exclusive use of ProcessBuilder.start() to
 * create new processes.
 *
 * @author Martin Buchholz
 * @version 1.32, 06/03/22
 * @since   1.5
 */

final class ProcessImpl extends Process {

    // System-dependent portion of ProcessBuilder.start()
    static Process start(String cmdarray[],
             java.util.Map<String,String> environment,
             String dir,
             boolean redirectErrorStream)
    throws IOException
    {
    String envblock = ProcessEnvironment.toEnvironmentBlock(environment);
    return new ProcessImpl(cmdarray, envblock, dir, redirectErrorStream);
    }

    private long handle = 0;
    private FileDescriptor stdin_fd;
    private FileDescriptor stdout_fd;
    private FileDescriptor stderr_fd;
    private OutputStream stdin_stream;
    private InputStream stdout_stream;
    private InputStream stderr_stream;

    private ProcessImpl(String cmd[],
            String envblock,
            String path,
            boolean redirectErrorStream)
    throws IOException
    {
    // Win32 CreateProcess requires cmd[0] to be normalized
    cmd[0] = new File(cmd[0]).getPath();

    StringBuilder cmdbuf = new StringBuilder(80);
    for (int i = 0; i < cmd.length; i++) {
            if (i > 0) {
                cmdbuf.append(' ');
            }
        String s = cmd[i];
        if (s.indexOf(' ') >= 0 || s.indexOf('\t') >= 0) {
            if (s.charAt(0) != '"') {
            cmdbuf.append('"');
            cmdbuf.append(s);
            if (s.endsWith("\\")) {
            cmdbuf.append("\\");
            }
            cmdbuf.append('"');
                } else if (s.endsWith("\"")) {
            /* The argument has already been quoted. */
            cmdbuf.append(s);
        } else {
            /* Unmatched quote for the argument. */
            throw new IllegalArgumentException();
        }
        } else {
            cmdbuf.append(s);
        }
    }
    String cmdstr = cmdbuf.toString();

    stdin_fd  = new FileDescriptor();
    stdout_fd = new FileDescriptor();
    stderr_fd = new FileDescriptor();

    handle = create(cmdstr, envblock, path, redirectErrorStream,
            stdin_fd, stdout_fd, stderr_fd);

    java.security.AccessController.doPrivileged(
        new java.security.PrivilegedAction() {
        public Object run() {
        stdin_stream =
            new BufferedOutputStream(new FileOutputStream(stdin_fd));
        stdout_stream =
            new BufferedInputStream(new FileInputStream(stdout_fd));
        stderr_stream =
            new FileInputStream(stderr_fd);
        return null;
        }
    });
    }

    public OutputStream getOutputStream() {
    return stdin_stream;
    }

    public InputStream getInputStream() {
    return stdout_stream;
    }

    public InputStream getErrorStream() {
    return stderr_stream;
    }

    public void finalize() {
    close();
    }

    public native int exitValue();
    public native int waitFor();
    public native void destroy();

    private native long create(String cmdstr,
                   String envblock,
                   String dir,
                   boolean redirectErrorStream,
                   FileDescriptor in_fd,
                   FileDescriptor out_fd,
                   FileDescriptor err_fd)
    throws IOException;

    private native void close();
}



那么提炼这段代码再打印出来,我们可以发现很大的问题是在 s.indexOf(' ') >= 0 这里,那么我们改下这段代码就能实现输出,代码如下


    /**
     * 模拟底层CMD实现输出查看命令行
     * @see ProcessImple.class
     * @author KARL-dujinyang
     * @param cmd
     * @return
     */
    public static String processImpl(String [] cmd){
        System.out.println("dujinyang start");
        StringBuilder cmdbuf = new StringBuilder(80);
        for (int i = 0; i < cmd.length; i++) {
                if (i > 0) {
                    cmdbuf.append(' ');
                }
            String s = cmd[i];
            if (s.indexOf(' ') >= 0 || s.indexOf('\t') >= 0) {
                if (s.charAt(0) != '"') {
                cmdbuf.append('"');
                cmdbuf.append(s);
                if (s.endsWith("\\")) {
                cmdbuf.append("\\");
                }
                cmdbuf.append('"');
                    } else if (s.endsWith("\"")) {
                /* The argument has already been quoted. */
                cmdbuf.append(s);
            } else {
                /* Unmatched quote for the argument. */
                throw new IllegalArgumentException();
            }
            } else {
                cmdbuf.append(s);
            }
        }
        System.out.println(cmdbuf.toString());
        System.out.println("dujinyang end");
    }



最后执行正确的String[] 数组,发现没什么问题,搞定!测试加上 "CMD.EXE", "/C" 的话,就会执行失败。

深圳市米奇云科技有限公司



执行命令的代码我也共享下吧 ~.~

深圳市米奇云科技


|| 版权声明:本文为博主杜锦阳原创文章,转载请注明出处。

如果有其它问题可留言或加入安卓移动技术精英群(246231638)

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以使用JavaRuntime类或ProcessBuilder类来执行cmd命令。例如,可以使用以下代码执行dir命令: ``` try { Process process = Runtime.getRuntime().exec("cmd /c dir"); BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream())); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } } catch (IOException e) { e.printStackTrace(); } ``` 这将在控制台输出当前目录下的文件和文件夹列表。 ### 回答2: Java中可以使用`Runtime`类和`ProcessBuilder`类来执行cmd命令。 使用`Runtime`类执行cmd命令的步骤如下: 1. 创建一个`Runtime`对象:`Runtime rt = Runtime.getRuntime();` 2. 使用`rt.exec("cmd命令")`来执行cmd命令,例如:`rt.exec("cmd /c dir");` 3. 通过`Process`对象获取执行结果并处理,例如可以使用`BufferedReader`读取`Process.getInputStream()`获取cmd命令执行的结果。 使用`ProcessBuilder`类执行cmd命令的步骤如下: 1. 创建一个`ProcessBuilder`对象:`ProcessBuilder pb = new ProcessBuilder("cmd", "/c", "cmd命令");` 2. 通过`ProcessBuilder`对象的`start()`方法启动cmd进程并执行命令:`Process process = pb.start();` 3. 通过`Process`对象获取执行结果并处理,例如可以使用`BufferedReader`读取`Process.getInputStream()`获取cmd命令执行的结果。 需要注意的是,执行cmd命令时可能会有一些限制,例如命令执行路径的权限问题。另外,还需要注意处理cmd命令执行过程中可能出现的异常情况。 ### 回答3: Java可以通过使用ProcessBuilder类或Runtime类来执行cmd命令。 使用ProcessBuilder类执行cmd命令的步骤如下: 1. 创建ProcessBuilder对象,传入要执行cmd命令作为参数。 2. 调用ProcessBuilder对象的start()方法,启动进程并执行命令。 3. 使用ProcessBuilder对象的waitFor()方法,等待进程执行完成。 4. 使用ProcessBuilder对象的getInputStream()方法获取命令执行的输出。 示例代码如下: ``` ProcessBuilder pb = new ProcessBuilder("cmd", "/c", "dir"); pb.redirectErrorStream(true); Process process = pb.start(); process.waitFor(); InputStream inputStream = process.getInputStream(); int ch; while ((ch = inputStream.read()) != -1) { System.out.print((char) ch); } ``` 上述代码中执行的是`dir`命令,通过循环读取InputStream的内容,可以获取到命令执行后返回的结果。 使用Runtime执行cmd命令的步骤如下: 1. 调用Runtime类的getRuntime()方法获取Runtime对象。 2. 调用Runtime对象的exec()方法,传入要执行cmd命令作为参数,返回一个Process对象。 3. 使用Process对象的waitFor()方法,等待进程执行完成。 4. 使用Process对象的getInputStream()方法获取命令执行的输出。 示例代码如下: ``` Runtime runtime = Runtime.getRuntime(); Process process = runtime.exec("cmd /c dir"); process.waitFor(); InputStream inputStream = process.getInputStream(); int ch; while ((ch = inputStream.read()) != -1) { System.out.print((char) ch); } ``` 上述代码中执行的同样是`dir`命令,通过循环读取InputStream的内容,可以获取到命令执行后返回的结果。 以上就是使用Java执行cmd命令的简单示例。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值