adb命令执行java代码

使用adb命令让Android设备执行java代码

参考:
http://blog.csdn.net/u010651541/article/details/53163542
http://blog.csdn.net/sahadev_/article/details/53318251

由于java是跨平台的,都是运行在java虚拟机中,而且Android也是使用Java开发,只不过Android app需要依赖Context环境,并且运行于Dalvik虚拟机或者ART虚拟机。
我就想:能不能像在电脑上一样,直接运行呢?于是搜索了一下,还真TM可以。

步骤一

编写java程序,假设文件名为Test.java(注意这里没有包名,后面会介绍有包名的情况),这里没什么好说的;

public class Test {
    public static void main(String[] args) {
        System.out.println("Android test");
    }
}

步骤二

编译Test.java并生成生成Test.class字节码文件,命令如下:
javac -source 1.7 -target 1.7 Test.java
编译完成以后,可以使用java Test命令运行一下.
这里为什么要指明为1.7呢?由于我使用的是jdk1.8,在步骤三使用dx命令的时候,会报unsupported class file version 52.0的错误.其中,file version 52.0对应的就是jdk1.8(参见https://en.wikipedia.org/wiki/Java_class_file) ,所有的对应关系如下:

Java SE 9 = 53 (0x35 hex),
Java SE 8 = 52 (0x34 hex),
Java SE 7 = 51 (0x33 hex),
Java SE 6.0 = 50 (0x32 hex),
Java SE 5.0 = 49 (0x31 hex),
JDK 1.4 = 48 (0x30 hex),
JDK 1.3 = 47 (0x2F hex),
JDK 1.2 = 46 (0x2E hex),
JDK 1.1 = 45 (0x2D hex).

步骤三

由于Android虚拟机(dalvik虚拟机或者ART)和pc上的java虚拟机有些不一样,所以.class字节码无法在Android设备上直接运行,这里使用Android SDK中的dx工具进行转换,dx命令生成Test.dex文件,命令如下:
dx --dex --output=Test.dex Test.class
Android studio的dx工具位于AndroidSDK/build-tools/(版本号)/目录下,建议添加到环境变量
--output=Test.dex指明.dex文件输出路径,可以是xxx\xxx\Test.dex

步骤四

push**步骤三**的Test.dex文件到手机(这里为了方便,直接push到了sdcard中)
push Test.dex /sdcard

步骤五

这里有两种方式运行Test.dex中的代码:
1. 直接使用app_process执行Test.dex文件
adb shell app_process -Djava.class.path=/sdcard/Test.dex /data/local/tmp Test
/data/local/tmp表示切换到/data/local/tmp目录下运行
Test表示start-class-name
2. 代码中new一个ClassLoader,并使用反射调用:

public Class loadClass(){
        //这里我直接把Test.dex文件push到sdcard了
        PathClassLoader pathClassLoader=new PathClassLoader("/sdcard/Test.dex", ClassLoader.getSystemClassLoader());
        try {
            //如果Test类存在包名,注意把包名加上eg:loadClass("com.example.Test");
            return pathClassLoader.loadClass("Test");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        return null;
    }
    public void invoke(Class cls){
        String[] strings=new String[1];
        try {
            //这里直接使用反射调用的main方法,如果存在其它方法,也可以使用反射来调用
            Method method=cls.getDeclaredMethod("main",String[].class);
            method.invoke(null,strings);
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }
    public void invoke(){
        Class cls=loadClass();
        if (cls!=null) {
            invoke(cls);
        }
    }

app_process参数格式:

app_process [vm-options] cmd-dir [options] start-class-name [main-options]

参数解释:

vm-options – VM 选项
cmd-dir –父目录 (/system/bin)
options –运行的参数 :
    –zygote
    –start-system-server
    –application (api>=14)
    –nice-name=nice_proc_name (api>=14)
start-class-name –包含main方法的主类  (com.android.commands.am.Am)
main-options –启动时候传递到main方法中的参数

java文件存在包名的情况

假设程序存在包名(这里假设包名为com.example),代码如下:

package com.example;

public class Test {
    public static void main(String[] args) {
        System.out.println("Android test");
    }
}
  1. 此时,使用java Test命令测试时,会出现如下错误:

    错误: 找不到或无法加载主类 Test

    执行这个命令的时候,我们应该位于…\src\com\example目录下,此时我们切换到…\src目录下执行:java com.example.Test即可;

  2. 同样,步骤三的dx命令,也存在类似问题,出现以下错误:

    PARSE ERROR:
    class name (com/example/Test) does not match path (Test.class)
    …while parsing Test.class
    1 error; aborting

    解决方案类似:切换到…\src目录下,执行dx --dex --output=com\example\Test.dex com\example\Test.class

  3. 执行命令改为:adb shell app_process -Djava.class.path=/sdcard/Test.dex /data/local/tmp com.example.Test

  4. 已经在步骤五代码注释中说明了

你可能感兴趣的:(Android)