利用DexClassLoader解决方法越界

背景:
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); }