DVM执行 java 程序的工具

1.  jvm 执行字节码原理:java 程序运行时,是由一个 java 虚拟机来解释 java 字节码的,它将这些字节码翻译成本地 CPU 的指令码,然后执行。

2.   Android 执行指令码原理:Android 应用程序打包成 dex 包后,通过系统程序 dalvikvm 创建一个虚拟机来执行参数中指定的 java 类。相对于 java 而言,负责解释并执行的就是一个虚拟机,而对于 Linux 而言,这就是一个普通的进程,它与一个只有一行代码的 Hello World 的可执行程序无本质区别。

3.   Android 启动一个虚拟机的方法跟启动任何一个可执行程序的方法是相同的,在命令行下输入可执行程序的名称,并在参数中指定要执行的 java 类即可。    

dalvikvm

dalvikvm 作用:创建一个虚拟机并执行指定的 java 类

dalvikvm 命令:dalvikvm -cp 文件路径 权限类名
如:
dalvikvm -cp /data/app/Demo.dex Demo

实例演示:

  • JVM 执行 java 程序的过程:

    1. 编译成二进制文件:javac Demo.java
    2. 翻译成机器码并执行:java Demo
  • DVM 执行 java 程序过程:对于 Android 而言,可执行代码需要转化成可执行的 dex 优化文件才能被系统加载执行。核心思想是将字节码文件转 dex 后,由 dvm 翻译执行。

    1. 打包为 jar 包: jar cvf Demo.jar Demo.class
    2. jar 转 dex:dx --dex --output= Demo1.jar Demo.jar
    3. 开一个 Android 模拟器,使用 Android 原生或 Genermotion 模拟器即可。
    4. 挂载设备:adb root;adb remount
    5. 安装程序:adb push Demo1.jar /data/app/Demo.dex
    6. 执行程序:adb shell dalvikvm -cp /data/app/Demo.dex Demo
  • DVM 执行 java 程序的过程中可能遇到的错误:

    • java sdk 和 dx 工具的要求版本不一致时,解决这种转换问题一般发生在第二步,比如我本地的 java 版本是 1.8.0_101-b13,可以使用 27.0.0 中的 build-tools 下的 dx 工具。

dvz

dvz 作用:从 zygote 进程中孵化出一个新的进程,新的进程也是一个 Dalvik 虚拟机。该进程与 dalvikvm 启动的虚拟机相比,区别在于该进程中已经预装了 Framework 中的大部分类和资源。

dvz 命令:dvz -classpath 文件路径 权限类名
如:
dvz -classpath /system/app/Demo.apk com.my.demo.DemoActivity

关于 Demo.apk,其代码如下:

上面的 main 函数并不是该程序的入口,只是用来作为开发调试。app 的主入口在 ActivityThread 中。

app_process

app_process 作用:Framework 启动过程中,加载 ZygoteInit.java 和 SystemServer.java,也可以用来调试 java 程序

app_process 命令:app_process -Djava.class.path=文件路径 路径 权限类名

如:app_process -Djava.class.path=/data/app/Demo.dex /data/app Demo

  • app_process 执行 java 程序的过程:

    1. 创建 Demo.java
    2. 执行 javac Demo.java
    3. 打包 jar 文件:java cvf Demo.jar Demo.class
    4. jar 转 dex:dx --dex --output=Demo1.jar Demo.jar
    5. 挂载设备:adb root;adb remount
    6. 将文件 push 到模拟器:adb push Demo1.jar /data/app/Demo.dex
    7. 执行命令:adb shell app_process -Djava.class.path=/data/app/Demo.dex /data/app Demo
  • app_process 命令参数:app_process [java-options] cmd-dir start-class-name [options]


由 app_process 想到的?

  • 系统命令封装:ampmwmsvc ...
  • 自定义命令封装:my_tool 适配不同平台工具差异。
  • 深度定制 zygoteframework,原因是:app_process 是初始化 zygote 的入口,属于安卓系统和Framework 启动的一个关键点。
  • 优化开机流程,减少开机过程耗时。

补充:

  • 补充1:
    • 执行 app_process,每次运行 Java 程序时,系统都会给其分配一个pid,并且进程名是app_process,通过追踪 pid 和 ppid,可以发现 fork 进程的大致流程为:/init --> /sbin/adbd -–> /system/bin/sh --> app_process,不同的安卓版本,会有差异,dalvik 和 art也不完全相同。例如在 Android 9.0 的设备上,流程则变为 /init --> zygote -->当前程序
  • 补充2:
    • app_process 启动的 Java 程序拥有shell级别的权限,所以像系统中的 ampmwmsvc 等程序才能执行。在 /system/bin 目录下执行:cat /system/bin/am 可以看到它其实并非一个二进制文件,而是一个可执行的 shell 脚本,由 app_process 执行了 com.android.commands.am.Am "$@",其执行原理是在当前窗口,将参数传递给 jvm,找到 Am 类的主函数,进行执行。

      console:/ # cat /system/bin/am
      #!/system/bin/sh
      
      if [ "$1" != "instrument" ] ; then
          cmd activity "$@"
      else
          base=/system
          export CLASSPATH=$base/framework/am.jar
          exec app_process $base/bin com.android.commands.am.Am "$@"
      fi
      console:/ #
      
    • 在 java 程序中执行 shellcmd 是可行的,分享一个可执行的案例,使用命令为:pm -l

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * 2019-05-18
 * java code for shell command demo, pm -l 
 */
public class DemoShellCmd {
    public static void main(String[] args) {
        System.out.println("");
        System.out.println("");
        System.out.println("DemoShellCmd::PMS 程序开始执行...");
        String cmd = "pm -l";
        try {
            Process exec = Runtime.getRuntime().exec(cmd);
            BufferedReader br = new BufferedReader(new InputStreamReader(exec.getInputStream()));
            String readLine=br.readLine();
            while(readLine!=null){
                System.out.println(readLine);
                readLine=br.readLine();
            }
            br.close();
            exec.destroy();
            exec=null;
            System.out.println("");
            System.out.println("");
            System.out.println("DemoShellCmd::PMS 程序执行完成");
        } catch (IOException e) {
            System.out.println("DemoShellCmd::PMS 程序执行异常");
            e.printStackTrace();
        }
    }
}
  • 补充3:
    • 进程状态如何查看: cat /proc/$pid/status 即可,其中 /proc 下遍布着系统运行过程中,所有进程 id 的信息,此文件夹下可以浏览你关注的进行状态,内存信息,线程信息...

猜你喜欢

转载自blog.csdn.net/lixiong0713/article/details/100514303