Android插件化——动态加载jar(一)

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插件化——动态加载jar(一)_第1张图片

3.2 Library中测试代码

这里就写个简单的测试方法,输入十位数和个位数,最后输出汇总的数据。
Android插件化——动态加载jar(一)_第2张图片

3.3 Library打包

这里打包aar,打jar包的方法网上一搜就很多
打开gradle 窗口,选择创建的library——other——assembleRelease方法,这样,在library——build——outputs——aar下面就会生成该release的aar包。
Android插件化——动态加载jar(一)_第3张图片
这里可以看下,打包的aar包中是否正确,包含测试方法。
Build——Analyze Apk ,选择library——build——outputs——aar下的library-release.aar包
Android插件化——动态加载jar(一)_第4张图片
Android插件化——动态加载jar(一)_第5张图片
这里可以看下打包后的代码,这里没做混淆,后面再说,在这可以看到,测试类已经包含,aar包打包正确。
Android插件化——动态加载jar(一)_第6张图片

3.4 主工程中加载

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

这里将aar包放在主工程的缓存目录下:
Android——data——gm.com.gui(包名)——cache文件夹下,没有cache,则自己手动创建下,等会读取aar路径也是该路径。
Android插件化——动态加载jar(一)_第7张图片

	/**
     * 加载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的下面给我评论就好。

你可能感兴趣的:(Android-studio)