1.动态加载jar的作用及意义
a.反破解,关于加密等比较隐蔽及隐私的的代码放在远端加载,即使别人破解了你的apk文件,也无法获取这些关键的代码,可以随时更换安全策略,提高应用的安全性;
b.间接解决方法数超过65535的限制,不再将jar打入apk文件中
c.可以进一步研究,拓展成apk的动态加载
2.原理
Dalvik虚拟机如同其他Java虚拟机一样,在运行程序时首先需要将对应的类加载到内存中。而在Java标准的虚拟机中,类加载可以从class文件中读取,也可以是其他形式的二进制流,因此,我们常常利用这一点,在程序运行时手动加载Class,从而达到代码动态加载执行的目的,然而Dalvik虚拟机毕竟不算是标准的Java虚拟机,因此在类加载机制上,它们有相同的地方,也有不同之处。我们必须区别对待,Dalvik虚拟机识别的是dex文件,而不是class文件。因此,我们供类加载的文件也只能是dex文件,或者包含有dex文件的.apk或.jar文件。
Android中和类加载相关的两个类:DexClassLoader和PathClassLoader
DexClassLoader :可以加载jar/apk/dex,也可以从SD卡中加载,通过其静态方法loadDex(path, outpath, 0) 得到DexFile对象;
PathClassLoader: 只能加载已经安装到Android系统中的apk文件,通过构造函数new DexFile(path)来产生DexFile对象。
这两者的区别在于DexClassLoader需要提供一个可写的outpath路径,用来释放.apk包或者.jar包中的dex文件。也就是说PathClassLoader不能主动从zip包中释放出dex,因此只支持直接操作dex格式文件,或者已经安装的apk(因为已经安装的apk在cache中存在缓存的dex文件), 而DexClassLoader可以支持.apk、.jar和.dex文件,并且会在指定的outpath路径释放出dex文件。
加载好类后,通常我们可以通过Java反射机制来使用这个类但是这样效率相对不高,而且也比较复杂凌乱。更好的做法是定义一个interface,并将这个interface写进容器端。待加载的类,继承自这个interface,并且有一个参数为空的构造函数,以使我们能够通过Class的newInstance方法产生对象然后将对象强制转换为interface对象,就可以直接调用成员方法了。
当你的jar中包含或者使用了第三方的类库你也可以定个规范的interface,并在你要加载的jar中实现这个interface,我们称这个实现为proxy类,通过proxy类调用你引用的第三方jar。我们将这个规范interface单独打成jar包放在主工程里,原理同上。
3.实现
创建一个规范interface,将其单独导出一个jar,我们叫interface jar,放在主工程及你要导入的jar项目中:
导出时只选这个interface就行了:
由于我们需要将第三方jar一起导出到一个jar中,还用这个方法导出的话我们无法将第三方jar一起导入,需要将引用的jar一起放入主工程才行,这违背了我们的初衷。这里我们用Idea编译器,新建一个项目DynamicLoadDemo,引用 interface jar,并实现ITest:
测试时我们引用 commons-io-2.2.jar 中的FileUtils,用来测试引用第三方jar。
将我们的DynamicLoadDemo导出成一个jar:
这时候我们导出的jar并不包含dex文件,需要我们用dx命令去处理,用命令行进入你的sdk->build-tools目录找到这个文件:
利用命令:
dx --dex --output=testjar1_temp.jar *\testJar1.jar
经过处理后你的testjar1_temp.jar 中包含dex文件就可以用DexClassLoader加载其中的类了
新建一个测试项目DynamicLoadJar(主工程),引入interface jar,并将testjar1_temp.jar放在你的手机sd卡中,
DexClassLoader(java.lang.String dexPath, java.lang.String optimizedDirectory, java.lang.String libraryPath, java.lang.ClassLoader parent)
参数意义:
java.lang.String dexPath, 需要装载的APK或者Jar文件的路径
java.lang.String optimizedDirectory, 优化后的dex文件存放目录,不能为null
java.lang.String libraryPath, 目标类中使用的C/C++库的列表, 可以为 null
java.lang.ClassLoader parent 该类装载器的父装载器,一般用当前执行类的装载器
执行结果:
完美!