最近项目有一个需求,需要根据用户需求动态加载APP内某功能模块,且当该部分功能模块代码有变动,只需更新该功能JAR,无需更新整个APK,基于这个需求,决定采用动态加载Jar的方式去实现,于是各种查资料进行验证,终于搞定,由于最近脑子容量不够,所以将实现方式整理出来并开放给大家进行参考。
实现过程需要注意的点:
1、当我们实现功能的Library工程中还依赖了其他第三方Jar,我们需要将其合并,否则你生成的Library对应的Jar包不含第三方文件会报错;
2、Android的虚拟机(Dalvik VM)是无法识别Java打出jar的class文件,DVM识别的是dex文件,所以需要通过dx工具转换成包含有dex文件的jar;
所以整个过程可以分为以下三个步骤来实现:
第一步:创建Library工程,导出Jar;
第二步:将主Jar和依赖Jar进行合并(ANT),将合并后的Jar进行DX处理;
第三步:利用DexClassLoader动态加载;
下面通过一个例子来介绍具体实现,首先看一下整个项目组成部分:
app:主工程Module;
commoninterfacelibrary:用于定义标准接口,供主项目集成,第三方实现具体功能;
dynamiclibrary:第三方实现功能库,在主项目中,动态加载该库;
utillibrary:工具库,模拟动态加载库中引用的第三方类库;
一、生成Jar
1、在commoninterfacelibrary中定义通用接口及方法:
定义完成后,生成common_interface.jar;
注:生成jar包方法比较简单,直接Build——>Make Project后,在build/intermediates/bundles/debug下找到classes.jar进行重命名。
2、将common_interface.jar拷贝到dynamiclibrary项目app/libs下,然后在dynamiclibrary下的gradle中添加对此jar的依赖:
定义ICommonInterface的实现类DynamicImp并实现具体方法:
注:MathUtil是定义在utillibrary下的一个工具类,生成出common_util.jar后拷贝到dynamiclibrary进行依赖,此处实现比较简单忽略该步。
按照上面生成jar的方法,生成出dynamiclibrary对应的dynamic.jar;
经过上述步骤后,我们得到三个jar包,分别是:
common_interface.jar 定义了标准接口;
common_util.jar 定义了工具类;
dynamic.jar 标准接口的实现类,我们需要动态加载的jar;
二、合并JAR并使用DX工具处理JAR包
下面我们开始介绍合并Jar的流程,由于common_interface.jar是标准接口,我们会在主项目中引用,所以不需要合并它,我们只需要合并common_util.jar和dynamic.jar。
对于合并Jar的方式,网上有很多方式,如fatjar插件、或者用IDEA工具,具体操作方法自行百度,我这里用的是ANT合并的方式,首先我们需要安装并配置ANT环境:
ANT下载官网地址:http://ant.apache.org/bindownload.cgi
下载完成后,解压,然后在我的电脑 –> 右键属性 –> 高级系统配置 -> 环境变量中配置ANT:
然后加入到系统变量的path中:
验证ANT环境是否已配置好:
看到上面的信息,证明ANT配置成功,下面来合并包,将需要合并的jar包放在同一个文件夹下,了解ant的都知道,ant构建的文件默认为build.xml,所以在文件夹下我们还需要创建该文件,并进行如下配置:
name : 表示的是你即将合成的jar包的名字 ;
basedir : 表示你存放jar包的目录;
上面配置文件的含义就是,将当前目录下所有jar进行合并,并在该目录下载生成名字为dynamic_combine.jar的新包,文件配置好后,执行下面的命令:
执行成功后,我们需要将该jar包使用dx工具进行转化:
将合并后的jar包拷贝到android-sdk/buildtools下的任意一个版本目录下,执行“dx --dex --output=dynamic_combine_dx.jar dynamic_combine.jar”
三、动态加载
执行成功后,我们得到最终的dynamic_combine_dx.jar,下面我们可以在主项目中进行动态加载了,通过两步来介绍动态加载过程:
1、下载dynamic_combine_dx.jar到SD卡:
首先将dynamic_combine_dx.jar拷贝到app/assets下(这里是为了模拟下载过程,实际项目中可配置到后台,从后台下载)
注:需在manifest添加权限
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
2、下载到SD卡成功后,进行动态加载调用相应方法:
先执行下载方法,后执行加载方法:
执行结果为:
成功,撒花!!!
网友评论
String dexPath = Environment.getExternalStorageDirectory().getAbsolutePath()
+ File.separator
+ "exp_dex.jar";
File file = new File(dexPath);
if(file.exists()){
Log.d("dml","name = "+ file.getName() + " size = " + file.length());
}else{
Log.d("dml","文件不存在");
return;
}
//内部路径
String dexInner = getDir("dex",0).getAbsolutePath();
DexClassLoader classLoader = new DexClassLoader(dexPath,dexInner,null,getClassLoader());
if(classLoader==null){
Log.d("dml","classLoader==null");
return;
}
try {
Class target = classLoader.loadClass("com.example.exp.ExpUtil");
Method method = target.getMethod("getName",null);
method.setAccessible(true);
String s = (String) method.invoke(target.newInstance(),null);
Log.d("dml","s = " + s);
} catch (ClassNotFoundException e) {
e.printStackTrace();
Log.d("dml","e = " + e.getMessage());
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}