背景:
Android工程方法数超过65535,则会提示编译错误(jar包太多)。为了减少jar包,可将一部分jar包转成dex文件(dex:andorid系统对jar的一些优化处理),dex文件在运行期间通过DexClassLoader加载至内存,从而避免方法数越界。
原理:
1.使用ClassLoader的好处:
扩充jar文件;
修改Framework中的已有类文件;
2.为什么使用DexClassLoader:
在Android中,ClassLoader是抽象类,一般使用DexClassLoader或者PathClassLoader加载,他们的区别是
DexClassLoader可以加载jar/apk/dex,可以从SD卡中加载未安装的apk
PathClassLoader只能加载系统中已经安装过的apk
源码,
public class DexClassLoader extends BaseDexClassLoader { public DexClassLoader(String dexPath, String optimizedDirectory,
String libraryPath, ClassLoader parent) { super(dexPath, new File(optimizedDirectory), libraryPath, parent); } }
public class PathClassLoader extends BaseDexClassLoader { public PathClassLoader(String dexPath, ClassLoader parent) { super(dexPath, null, null, parent); } public PathClassLoader(String dexPath, String libraryPath,
ClassLoader parent) { super(dexPath, null, libraryPath, parent); } }
- dexPath: 指目标类所在的jar/apk文件路径, 多个路径使用
File.pathSeparator分隔, Android里面默认为 “:” - optimizedDirectory:
解压出的dex文件的存放路径,以免被注入攻击,不可存放在外置存储。(DexClassLoader 的optimizedDirectory不能为空) - libraryPath :目标类中的C/C++库存放路径。
- parent: 父类装载器
3.生成所需dex包
cmd至sdk解压目录…\AndroidSDK\build-tools\android-4.4W,
将jar文件转换成dex二进制jar文件:dx –dex –output=classes.dex libs(libs是要转换的jar文件总目录)
4.通过DexClassLoader加载生成的dex文件
- 将classes.dex放在工程的asserts资源目录下(或者手机内部存储)
- 生成dex文件输出流
在自定义Application中,重写attachBaseContext,插入
String fileName = "classes.dex";
String internalPath = context.getExternalCacheDir() + File.separator + "classes.dex";
File desFile = new File(internalPath);
InputStream in = null;
OutputStream out = null;
try {
in = context.getApplicationContext().getAssets().open(fileName);
out = new FileOutputStream(desFile.getAbsolutePath());
byte[] bytes = new byte[1024];
int i;
while ((i = in.read(bytes)) != -1)
out.write(bytes, 0 , i);
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
if (in != null)
in.close();
if (out != null)
out.close();
} catch (IOException e) {
e.printStackTrace();
}
}
- 设置输出流的类加载器
try
{
Field field = ClassLoader.class.getDeclaredField("parent");
field.setAccessible(true);
ClassLoader classLoader = context.getApplication().getClassLoader();
//设置dexclassloader解析的文件目录
File dexdst = new File(desFile, "dst");
dexdst.mkdir();
DexClassLoader dexClassLoader = new DexClassLoader(desFile.getAbsolutePath(), dexdst.getAbsolutePath(),
context.getApplication().getApplicationInfo().nativeLibraryDir, classLoader.getParent());
field.set(classLoader, dexClassLoader);
} catch (Exception e)
{
throw new RuntimeException(e);
}