参考:
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 [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方法中的参数
假设程序存在包名(这里假设包名为com.example),代码如下:
package com.example;
public class Test {
public static void main(String[] args) {
System.out.println("Android test");
}
}
此时,使用java Test
命令测试时,会出现如下错误:
错误: 找不到或无法加载主类 Test
执行这个命令的时候,我们应该位于…\src\com\example目录下,此时我们切换到…\src目录下执行:java com.example.Test
即可;
同样,步骤三的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
执行命令改为:adb shell app_process -Djava.class.path=/sdcard/Test.dex /data/local/tmp com.example.Test