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

简介: 奥特曼超人杜锦阳曾经说过:“宁可在法度外灭亡,不在法度中生存。” Cmd命令执行失败,可能大家开发中经常会有遇到如下问题,可是百度谷歌却出不来,博主踩的坑共享给大家:报错 CreateProcess error=2, ϵͳÕҲ»µ½ָ¶ JAVA ...

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

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



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)

相关文章
|
3月前
|
存储 Java
【编程基础知识】 分析学生成绩:用Java二维数组存储与输出
本文介绍如何使用Java二维数组存储和处理多个学生的各科成绩,包括成绩的输入、存储及格式化输出,适合初学者实践Java基础知识。
99 1
|
14天前
|
缓存 算法 搜索推荐
Java中的算法优化与复杂度分析
在Java开发中,理解和优化算法的时间复杂度和空间复杂度是提升程序性能的关键。通过合理选择数据结构、避免重复计算、应用分治法等策略,可以显著提高算法效率。在实际开发中,应该根据具体需求和场景,选择合适的优化方法,从而编写出高效、可靠的代码。
25 6
|
2月前
|
监控 算法 Java
jvm-48-java 变更导致压测应用性能下降,如何分析定位原因?
【11月更文挑战第17天】当JVM相关变更导致压测应用性能下降时,可通过检查变更内容(如JVM参数、Java版本、代码变更)、收集性能监控数据(使用JVM监控工具、应用性能监控工具、系统资源监控)、分析垃圾回收情况(GC日志分析、内存泄漏检查)、分析线程和锁(线程状态分析、锁竞争分析)及分析代码执行路径(使用代码性能分析工具、代码审查)等步骤来定位和解决问题。
|
2月前
|
Java Shell Windows
java Runtime.exec()执行shell/cmd命令:常见的几种陷阱与一种完善实现
java Runtime.exec()执行shell/cmd命令:常见的几种陷阱与一种完善实现
55 1
|
3月前
|
存储 SQL 小程序
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
这篇文章详细介绍了Java虚拟机(JVM)的运行时数据区域和JVM指令集,包括程序计数器、虚拟机栈、本地方法栈、直接内存、方法区和堆,以及栈帧的组成部分和执行流程。
47 2
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
|
2月前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
72 2
|
2月前
|
Java 关系型数据库 数据库
面向对象设计原则在Java中的实现与案例分析
【10月更文挑战第25天】本文通过Java语言的具体实现和案例分析,详细介绍了面向对象设计的五大核心原则:单一职责原则、开闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则帮助开发者构建更加灵活、可维护和可扩展的系统,不仅适用于Java,也适用于其他面向对象编程语言。
46 2
|
3月前
|
Java Windows
JAVA 常用的 DOS 命令
【10月更文挑战第15天】DOS 命令是 Java 开发中不可或缺的工具,掌握这些命令可以提高开发效率和操作便利性。
46 3
|
3月前
|
Java
让星星⭐月亮告诉你,Java synchronized(*.class) synchronized 方法 synchronized(this)分析
本文通过Java代码示例,介绍了`synchronized`关键字在类和实例方法上的使用。总结了三种情况:1) 类级别的锁,多个实例对象在同一时刻只能有一个获取锁;2) 实例方法级别的锁,多个实例对象可以同时执行;3) 同一实例对象的多个线程,同一时刻只能有一个线程执行同步方法。
24 1
|
3月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
62 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用