美文网首页Android专栏程序员Android开发常见小问题列表
Android动态加载Jar(包含第三方依赖Jar)

Android动态加载Jar(包含第三方依赖Jar)

作者: Hank802 | 来源:发表于2017-10-23 17:01 被阅读199次

            最近项目有一个需求,需要根据用户需求动态加载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卡成功后,进行动态加载调用相应方法:

    先执行下载方法,后执行加载方法:

    执行结果为:

    成功,撒花!!!

    相关文章

      网友评论

      • 08_carmelo://dex外部路经
        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();
        }
      • 08_carmelo:一直报错: Didn't find class "com.example.exp.ExpUtil" on path: DexPathList[[zip file "/storage/emulated/0/exp_dex.jar"],nativeLibraryDirectories=[/system/lib64, /system/vendor/lib64]] ,并且我看dex并没有被从sd卡拷贝到data/app/0那个目录, 哪里出问题了?

      本文标题:Android动态加载Jar(包含第三方依赖Jar)

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