Android插件化——动态加载jar,apk,dex,aar(一)


文章目录

  • Android插件化——动态加载jar,apk,dex,aar(一)
  • 1.类加载机制
  • 2.DexClassLoader
  • 3.Demo演示
  • 3.1 新建工程及Lib
  • 3.2 Library中测试代码
  • 3.3 Library打包
  • 3.4 主工程中加载
  • 3.5 最后
  • 4.结束


1.类加载机制

在 Android 中,App 安装到手机后,app每次启动一个进程,Android虚拟机(Dalvik VM)运行读取apk里面的dex文件,apk 里面的 class.dex 中的 class 均是通过PathClassLoader 来加载的。除此之外,Android还提供了 DexClassLoader 可以用来加载 SD 卡上加载包含 class.dex 的 .jar 和 .apk 文件。

DexClassLoader 和 PathClassLoader 的都是继承与 BaseDexClassLoader,是通过类加载 ClassLoader 来加载查找 class。PathClassLoader只能加载已经安装到Android系统中的apk文件,DexClassLoader可以加载外部(如SD卡)的jar/apk/dex/aar。

2.DexClassLoader

public class DexClassLoader extends BaseDexClassLoader {
  public DexClassLoader(String dexPath, String optimizedDirectory, String librarySearchPath, ClassLoader parent) {
    super((String)null, (File)null, (String)null, (ClassLoader)null);
    throw new RuntimeException("Stub!");
  }
}

参数意义
dexPath: 需要加载的APK或者Jar文件的路径,绝对路径。

optimizedDirectory: 解压后的dex文件存放目录,不能为null,可以设置为getCacheDir()

libraryPath: 目标类中使用的C/C++库的列表, 可以为 null

parent : 该类装载器的父装载器,一般用当前执行类的装载器getClassLoader()

3.Demo演示

3.1 新建工程及Lib

新建一个工程及Library.。具体的就不演示了,最终效果如下图。

android动态加载sdk android动态加载jar_Android

3.2 Library中测试代码

这里就写个简单的测试方法,输入十位数和个位数,最后输出汇总的数据。

android动态加载sdk android动态加载jar_android_02

3.3 Library打包

这里打包aar,打jar包的方法网上一搜就很多

打开gradle 窗口,选择创建的library——other——assembleRelease方法,这样,在library——build——outputs——aar下面就会生成该release的aar包。

android动态加载sdk android动态加载jar_Android_03


这里可以看下,打包的aar包中是否正确,包含测试方法。

Build——Analyze Apk ,选择library——build——outputs——aar下的library-release.aar包

android动态加载sdk android动态加载jar_android_04


android动态加载sdk android动态加载jar_android动态加载sdk_05


这里可以看下打包后的代码,这里没做混淆,后面再说,在这可以看到,测试类已经包含,aar包打包正确。

android动态加载sdk android动态加载jar_加载_06

3.4 主工程中加载

将上一步生成的aar包放在手机sd卡的可读取路径下,然后测试调用aar包中的方法。

这里将aar包放在主工程的缓存目录下:

Android——data——gm.com.gui(包名)——cache文件夹下,没有cache,则自己手动创建下,等会读取aar路径也是该路径。

android动态加载sdk android动态加载jar_android动态加载sdk_07

/**
     * 加载dex文件中的class,并调用其中的方法
     * 这里由于是加载 jar文件,所以采用DexClassLoader 
     * 下面开始加载dex class
     */
    public static void loadDexClass(Context context) {
        File cacheFile = FileUtils.getCacheDir(context);
        String internalPath = cacheFile.getAbsolutePath() + File.separator + "library-release.aar";
        File desFile = new File(internalPath);
        if (desFile.exists()) {
            DexClassLoader dexClassLoader = new DexClassLoader(internalPath, cacheFile.getAbsolutePath(), null, context.getClass().getClassLoader());
            try {
                Class libClazz = dexClassLoader.loadClass("gm.com.library.GuiTest");
                Constructor<?> localConstructor = libClazz.getConstructor();
//
                Object obj = localConstructor.newInstance();
                Method mMethodWrite = libClazz.getMethod("get",int.class,int.class);
                mMethodWrite.setAccessible(true);
                Integer  str = (Integer ) mMethodWrite.invoke(obj,2,2);
                Toast.makeText(context,"result is " + str,Toast.LENGTH_SHORT).show();

            } catch (Exception e) {
                Toast.makeText(context,"result is error " + e.getMessage(),Toast.LENGTH_SHORT).show();
            }
        }
    }

3.5 最后

最后,不要忘记增sd卡的读取权限。6.0及以上的自己在代码中增加sd卡权限。

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

ps:有时由于编译问题,导致dexClassLoader.loadClass报NotFound异常,因此在主工程和library的build.gradle文件中的android中加上compileOptions 方法,防止编译报错。

android {
    compileSdkVersion 29
    defaultConfig {
        ......
    }

    buildTypes {
       ......
    }

    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }

}

4.结束

最后,代码没有上传,有需要demo的下面给我评论就好。