Android 中动态加载.jar的实现步骤
程序员文章站
2023-12-01 08:13:16
首先第一个是 jar 文件的制作,java 里面直接把 .class 文件打包到 .jar 文件里面就可以了,但是 android 的 dalvik vm 是不认 java...
首先第一个是 jar 文件的制作,java 里面直接把 .class 文件打包到 .jar 文件里面就可以了,但是 android 的 dalvik vm 是不认 java 的 byte code 的,所以不能直接这么打包,而要用 dx 工具转成 dalvik byte code 才可以。当然,dx 工具转了之后,jar 包里面就不 是 .class 文件了,而是 .dex 文件。
第二个是,android 里面虽然也提供了 urlclassloader 的实现,但是并不能用。要动态加载其它类,可以用的 class loader 有:
dexclassloader,pathclassloader,其中,dexclassloader 可以加载 apk, jar 或者 dex 文件,例如:
file jarfile = new file("/sdcard/test.dex");
# if ( jarfile.exists() ) {
dexclassloader cl = new dexclassloader(jarfile.tostring(), "/sdcard/test", null, classloader.getsystemclassloader());
class<?> c = cl.loadclass("xiaogang.test.test");
但是 dexclassloader 要求指定一个可写的目录,即 dexclassloader 构造函数的第二个参数,在上例中是 /sdcard/test
这个参数的含义是:directory where optimized dex files should be written
因为 dalvik 在加载 dex 文件时,会动态进行优化,dexclassloader 要求指定优化后 dex 文件存放的位置。
pathclassloader 的限制要更多一些,它只能加载已经安装到 android 系统中的 apk 文件,也就是 /data/app 目录下的 apk 文件。其它位置的文件加载的时候都会出现 classnotfoundexception. 例如:
pathclassloader cl = new pathclassloader(jarfile.tostring(), "/data/app/" , classloader.getsystemclassloader());
由于 pathclassloader 会去读取 /data/dalvik-cache 目录下的经过 dalvik 优化过的 dex 文件,这个目录的 dex 文件是在安装 apk 包的时候 由 dalvik 生成的。例如,如果包的名字是 xiaogang.test,android 应用安装之后都保存在 /data/app 目 录下,即 /data/app/xiaogang.test-1.apk,那么 /data/dalvik-cache 目录下就会生成 data@app@xiaogang.test-1.apk@classes.dex 文件。在调用 pathclassloader 时,它就 会按照这个规则去找 dex 文件,如果你指定的 apk 文件是 /sdcard/test.apk,它按照这个规则就会去读 /data /dalvik-cache/sdcard@test.apk@classes.dex 文件,显然这个文件不会存在,所 以pathclassloader 会报错。
在 google 修正这个问题之前,我们要么就只能用 dexclassloader,要么就只能用 pathclassloader 加载已安装的 apk 。
第二个是,android 里面虽然也提供了 urlclassloader 的实现,但是并不能用。要动态加载其它类,可以用的 class loader 有:
dexclassloader,pathclassloader,其中,dexclassloader 可以加载 apk, jar 或者 dex 文件,例如:
复制代码 代码如下:
file jarfile = new file("/sdcard/test.dex");
# if ( jarfile.exists() ) {
dexclassloader cl = new dexclassloader(jarfile.tostring(), "/sdcard/test", null, classloader.getsystemclassloader());
class<?> c = cl.loadclass("xiaogang.test.test");
但是 dexclassloader 要求指定一个可写的目录,即 dexclassloader 构造函数的第二个参数,在上例中是 /sdcard/test
这个参数的含义是:directory where optimized dex files should be written
因为 dalvik 在加载 dex 文件时,会动态进行优化,dexclassloader 要求指定优化后 dex 文件存放的位置。
pathclassloader 的限制要更多一些,它只能加载已经安装到 android 系统中的 apk 文件,也就是 /data/app 目录下的 apk 文件。其它位置的文件加载的时候都会出现 classnotfoundexception. 例如:
复制代码 代码如下:
pathclassloader cl = new pathclassloader(jarfile.tostring(), "/data/app/" , classloader.getsystemclassloader());
由于 pathclassloader 会去读取 /data/dalvik-cache 目录下的经过 dalvik 优化过的 dex 文件,这个目录的 dex 文件是在安装 apk 包的时候 由 dalvik 生成的。例如,如果包的名字是 xiaogang.test,android 应用安装之后都保存在 /data/app 目 录下,即 /data/app/xiaogang.test-1.apk,那么 /data/dalvik-cache 目录下就会生成 data@app@xiaogang.test-1.apk@classes.dex 文件。在调用 pathclassloader 时,它就 会按照这个规则去找 dex 文件,如果你指定的 apk 文件是 /sdcard/test.apk,它按照这个规则就会去读 /data /dalvik-cache/sdcard@test.apk@classes.dex 文件,显然这个文件不会存在,所 以pathclassloader 会报错。
在 google 修正这个问题之前,我们要么就只能用 dexclassloader,要么就只能用 pathclassloader 加载已安装的 apk 。