Android动态加载jar(jar中包含第三方jar)

作者: 帝都大勇哥 | 来源:发表于2016-03-15 18:06 被阅读1941次

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规范

导出时只选这个interface就行了:

只导出这个ITest

由于我们需要将第三方jar一起导出到一个jar中,还用这个方法导出的话我们无法将第三方jar一起导入,需要将引用的jar一起放入主工程才行,这违背了我们的初衷。这里我们用Idea编译器,新建一个项目DynamicLoadDemo,引用 interface jar,并实现ITest:

测试时我们引用 commons-io-2.2.jar 中的FileUtils,用来测试引用第三方jar。

将我们的DynamicLoadDemo导出成一个jar:

选取我们要导出的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  该类装载器的父装载器,一般用当前执行类的装载器

执行结果:

完美!

相关文章

网友评论

  • ab2f877773f7:怎么将第三方jar打入自己jar包的
    帝都大勇哥:@ab2f877773f7 AS我试过,可以导出Jar包但是不能导出依赖的第三方Jar包,我试的时候是Java 工程,Android 的话你也可以做个尝试,按道理也是可行的
    ab2f877773f7:@帝都大勇哥 AS 实现不了吗。另外,是用IDEA建一个Android 工程还是Java工程呢
    帝都大勇哥:@ab2f877773f7 里面有说,导出Jar 的时候可以选择依赖的Jar,IDEA编译器
  • 毹毹:必须要在ec上面打jar么?
    帝都大勇哥:不是的,其他编译器也可以
  • shawn_h:前面写的很好,后面有点模糊了

本文标题:Android动态加载jar(jar中包含第三方jar)

本文链接:https://www.haomeiwen.com/subject/tqcxlttx.html