在 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。
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()
新建一个工程及Library.。具体的就不演示了,最终效果如下图。
这里就写个简单的测试方法,输入十位数和个位数,最后输出汇总的数据。
这里打包aar,打jar包的方法网上一搜就很多
打开gradle 窗口,选择创建的library——other——assembleRelease方法,这样,在library——build——outputs——aar下面就会生成该release的aar包。
这里可以看下,打包的aar包中是否正确,包含测试方法。
Build——Analyze Apk ,选择library——build——outputs——aar下的library-release.aar包
这里可以看下打包后的代码,这里没做混淆,后面再说,在这可以看到,测试类已经包含,aar包打包正确。
将上一步生成的aar包放在手机sd卡的可读取路径下,然后测试调用aar包中的方法。
这里将aar包放在主工程的缓存目录下:
Android——data——gm.com.gui(包名)——cache文件夹下,没有cache,则自己手动创建下,等会读取aar路径也是该路径。
/**
* 加载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();
}
}
}
最后,不要忘记增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
}
}
最后,代码没有上传,有需要demo的下面给我评论就好。