美文网首页插件开发Android SmallAndroid开发
Small插件化源码分析--ReflectAccelerator

Small插件化源码分析--ReflectAccelerator

作者: CangWang | 来源:发表于2016-11-29 12:53 被阅读281次

    我系苍王,欢迎关注的源码分析的第五节。

    欢迎浏览我之前的文章,有兴趣可以参考一下,可以给个喜欢或者关注我的文章,谢谢。

    [Android]如何做一个崩溃率少于千分之三噶应用app(9)-Small插件化

    Small插件化源码分析--启动流程

    Small插件化源码分析--热更流程

    Small插件化源码分析--InstumentationWrapper生命周期

    五.ReflectAccelerator的奏曲

    这一节,我们要分析Small真正实现的核心代码

    这个文件就是ReflectAccelerator的代码。

    其相关到resource,dex,lib替换的相关流程

    1.这里需要大家对classloader有一个基本的认识

    Dalvik虚拟机如同其他Java虚拟机一样,在运行程序时首先需要将对应的类加载到内存中。而在Java标准的虚拟机中,类加载可以从class文件中读取,也可以是其他形式的二进制流。

      PathClassLoader:它只能加载已经安装到 Android 系统中的 apk 文件,也就是 /data/app 目录下的 apk 文件

      DexClassLoader: 可以加载 apk, jar 或者 dex 文件

    而插件化真正用到的就是DexClassLoader,其继承于ClassLoader。

    源码在DexClassLoader源码,其源码在dalvik虚拟机的目录里面,所以platform里面无法看到全部的源码。

    我们从第一节启动流程中有一个初略的说明,还是分别从resource,dex,lib的替换流程来分析吧

    (1)我们首先是利用mergeResouce这个函数来加载资源resource的

    我们需要自己新建一个AssetManager或者获取Aplication的AssetManager。

    addAssetPaths中,通过反射的方式来把资源路径和assetManager替换成我们自己定义的资源

    ensureStringBlocks来创建字符串资源池

    新建弱引用集合Resources资源

    4.4以上的版本,通过反射一个新的ResourcesManager,用数组表获取现在启动应用的资源列表。如果获取不了会取出reourcesManager里面的mResourceReferences资源指向

    4.4以下的版本,使用哈希表来获取mActiviteResources活跃资源

    通过遍历资源Resources,来替换newAssetManger到mAssets属性里面

    异常会反射mResourcesImpl代理接口,来替换到mAssets里面

    替换后,触发资源的配置刷新,让其可以生效

    如果是5.0以上的系统,就需要清除应用资源每个字节池里面的内容。

    Small里面,并没有提供移除资源的方法,如果需要改写,可以自己去添加一个方法。

    (2)添加Dex的时候,使用的是expandDexPathList的方法

    其有少于9到13可以调用的方法,还有4.0以上所调用的方法

    先看一下9到13的方法

    首先获取mFiles mPaths mZips mDexs的属性

    替换掉全部的这些属性的变量

    通过expandArray调用System.arraycopy,来添加dexclassloader里面的这些属性值,

    当push为true时,其排序是先将目标的长度的elements写入,再写入原来已经加载好的elements到后面

    当push为false时,是先将已有elements写入,然后将自己的写到已有的之后

    这里和MultiDex.java使用的是类似的替换方法

    这里System.arraycopy的方法是会调用到native层去运行的,从参数可以看出其运行规则

    再看一下Android4.0以上的添加Dex方法,

    其需要的是将Elemets的变量封装一个Object的数组里面

    Android4.0以上有DexPathList.java里面封装了一个Element的类,用于保存Dex的信息

    其makeDexElement的方法,就是用来封装Element的表

    获取DexPathsList的Delement的类来动态新建一个Element。

    使用fillDexPathList来反射pathList和dexElements的属性,再使用expandArray来添加资源到尾部

    (3)通过expandNativeLibraryDirectories来添加lib的.so文件

    android 9~13中,只是获取libraryPathDeements的属性,然后获取加载路径列表,然后把自定义的列表添加到原来的列表里面,需要用“/”来分隔

    android14~22中,是使用反射nativeLibraryDirectories的接口,然后通过expandArray来写入到library列表之后

    android23以上,稍显复杂,需要先索引到nativeLibraryDirectories目录,然后再获取natveLibraryPathElements的Element对象列表,然后通过封装Element的列表对象,然后再通过expandArray放到末尾

    其加载的方法就到这里,然而RefectAccelerator还有一些比较特殊的方法

    获取ActvityThread,如果currentActivityThread存在就返回这个属性,否则通过读取mLoadedApk里面的mActivityThread来获取

    上一节InstrumentationWrapper的execStartActivity实际上也是通过ReflectAccelerator封装的

    android 9~20 包装好types,然后通过getMethod反射获取execStartActivity的方法,然后再通过反射再次自己的属性值

    反射获取方法

    android21以上原理和以上的以上,只是参数不同而已

    *****************这里是最后的福利******************

    这里提供了sliceArray的方法,这个方法是用来删除array里面的某个项的资源的。这个有什么用,这个对对自定义删除dex,library都起到至关重要的作用,如果利用这个源码就得大家深入去探究啦。我这边已经是完成了删除resource,dex,library的源码修改。

    这节课程就到这里。

    Small源码分析暂时就到这里。

    如果以后有时间会再去查看一下aapt等资源打包的流程。

    下一节,敬请期待!!!

    相关文章

      网友评论

        本文标题:Small插件化源码分析--ReflectAccelerator

        本文链接:https://www.haomeiwen.com/subject/gfiupttx.html