Replugin 全面解析 (2)

Activity作为四大组件中最重要的组件,在Replugin中对它的支持的架构设计也是最复杂的,所以本篇分析我们就来看看Activity的启动流程。

以下这张图简要的画出类Activity启动的过程,当然简化了一些流程:

  • Pmbase根据Intent找到对应的插件
  • 分配坑位Activity,与插件中的Activity建立一对一的关系并保存在PluginContainer
  • 让系统启动坑位Activity,因为它是在Manifest中注册过的
  • Android系统会尝试使用RepluginClassLoader加载坑位ActivityClass对象
  • RepluginClassLoader 通过建立的对应关系找到插件Activity,并使用PluginDexClassLoader 加载插件Activity 的Class对象并返回
  • Android系统就使用这个插件中的Activity的Class对象来运行生命周期函数

Android系统就是这样被欺骗了!

activity.jpg

启动一个Activity的入口函数是Replugin.startActivity(),然后调用Factory.startActivityWithNoInjectCN,再经过PluginCommImpl.startActivivty(),最终来到PluginLibraryInternalProxy.startActivity(),这里将是真正开始工作的地方,会分为以下几个步骤:

  • 如果有必要,需要先下载插件

    下载过程会通过回调让用户去实现,比如显示进度,安装等。

    if (download) {
        if (PluginTable.getPluginInfo(plugin) == null) {
            if (isNeedToDownload(context, plugin)) {
                return RePlugin.getConfig().getCallbacks().onPluginNotExistsForActivity(context, plugin, intent, process);
            }
        }
    }
    
  • 检查插件状态

    如果插件状态不正确,或者首次加载大插件,会通过回调让用户处理,用户可以可以在回调里定制自己的行为,比如弹出提示框,加载进度条等。

    if (PluginStatusController.getStatus(plugin) < PluginStatusController.STATUS_OK) {
        return RePlugin.getConfig().getCallbacks().onPluginNotExistsForActivity(context, plugin, intent, process);
    }
    
    if (!RePlugin.isPluginDexExtracted(plugin)) {
        PluginDesc pd = PluginDesc.get(plugin);
        if (pd != null && pd.isLarge()) {
            return RePlugin.getConfig().getCallbacks().onLoadLargePluginForActivity(context, plugin, intent, process);
        }
    }
    
  • 寻找坑位,启动坑位Activity

    调用在PluginLibraryInternalProxy.startActivity()中调用PluginCommImpl.loadPluginActivity来寻找坑位Activity。

    请注意注释中的分支C,如果是第一次去获取信息,会首先去加载插件的Dex文件以及资源等,并创建PluginDexClassLoader。这个分支我们在后面来讲解。

    public ComponentName loadPluginActivity(Intent intent, String plugin, String activity, int process) {
        ActivityInfo ai = null;
        String container = null;
        PluginBinderInfo info = new PluginBinderInfo(PluginBinderInfo.ACTIVITY_REQUEST);
    
        try {
            ai = getActivityInfo(plugin, activity, intent); //分支C:获取 ActivityInfo
            // 根据 activity 的 processName,选择进程 ID 标识
            if (ai.processName != null) {
                process = PluginClientHelper.getProcessInt(ai.processName);
            }
            // 容器选择(启动目标进程,如果有必要的话,一般默认会使用UI进程)
            IPluginClient client = MP.startPluginProcess(plugin, process, info);
          ......
            // 远程分配坑位
            container = client.allocActivityContainer(plugin, process, ai.name, intent);
        } catch (Throwable e) {
        }
    
        PmBase.cleanIntentPluginParams(intent);
      ......
        return new ComponentName(IPC.getPackageName(), container);
    }
    

    来重点看看坑位分配,这是一个远程调用,调用了Persistent进程中的PluginProcessPer.allocActivityContainer函数,进一步调用bindActivity函数。

    final String bindActivity(String plugin, int process, String activity, Intent intent) {
        Plugin p = mPluginMgr.loadAppPlugin(plugin); //获取插件对象
      ......
        ActivityInfo ai = p.mLoader.mComponents.getActivity(activity); //获取ActivityInfo
      ......
        String container;
        // 自定义进程
        if (ai.processName.contains(PluginProcessHost.PROCESS_PLUGIN_SUFFIX2)) {
            String processTail = PluginProcessHost.processTail(ai.processName);
            container = mACM.alloc2(ai, plugin, activity, process, intent, processTail);
        } else {
            container = mACM.alloc(ai, plugin, activity, process, intent);
        }
      ......
        return container;
    }
    

    这里mACM.alloc2调用allocLocked函数真正的执行了坑位分配的任务。这段代码简单明了,注意坑位找好以后会返回一个AcitivtyState对象,这里面保存了坑位Activity和真实要启动的Activity之间的对应关系。并且这个对应关系会被保存起来,在RepluginClassLoader在加载类的时候会被拿出来使用,以获取要运行的Activityclass对象。

    private final ActivityState allocLocked(ActivityInfo ai, HashMap<String, ActivityState> map, String plugin, String activity, Intent intent) {
        // 首先找上一个活的,或者已经注册的,避免多个坑到同一个activity的映射
        for (ActivityState state : map.values()) {
            if (state.isTarget(plugin, activity)) {
                return state;
            }
        }
        // 新分配:找空白的,第一个
        for (ActivityState state : map.values()) {
            if (state.state == STATE_NONE) {
                state.occupy(plugin, activity);
                return state;
            }
        }
        ActivityState found;
        // 重用:则找最老的那个
        found = null;
        for (ActivityState state : map.values()) {
            if (!state.hasRef()) {
                if (found == null) {
                    found = state;
                } else if (state.timestamp < found.timestamp) {
                    found = state;
                }
            }
        }
        if (found != null) {
            found.occupy(plugin, activity);
            return found;
        }
        // 强挤:最后一招,挤掉:最老的那个
        found = null;
        for (ActivityState state : map.values()) {
            if (found == null) {
                found = state;
            } else if (state.timestamp < found.timestamp) {
                found = state;
            }
        }
        if (found != null) {
            found.finishRefs();
            found.occupy(plugin, activity);
            return found;
        }
        return null;
    }
    

    坑位找到啦!PluginLibraryInternalProxy.startActivity()中开始启动坑位Activity,就在分支D的位置,这个分支我们稍微延后一点来展开。

    public boolean startActivity(Context context, Intent intent, String plugin, String activity, int process, boolean download) {
        ......
        ComponentName cn = mPluginMgr.mLocal.loadPluginActivity(intent, plugin, activity, process);  // 找到坑位组件
        ......
        // 将Intent指向到“坑位”
        intent.setComponent(cn);
        ......
        context.startActivity(intent);  //分支D: 启动坑位Activity
    
        return true;
    }
    

    看到这里你一定会疑惑,难道这样插件的Activity就启动起来啦吗?这启动的明明就是一个坑位Activity啊?别着急,接着就是前面一直强调的唯一hook点发挥作用的时候啦!!

  • Dex的加载以及Activity的加载启动

    上面有一个分支C你还记得吗?我们将它与Activity的加载流程放在一起来讲,因为这两者是紧密相关的。

    先来看分支C。

    • PluginCommImpl.getActivityInfo调用PmBase.loadAppPlugin获取插件对象,从下面注释的分支可以看出,Replugin 是支持使用 IntentFilter 来启动组件的,完美支持原生特性,是不是很赞!

      public ActivityInfo getActivityInfo(String plugin, String activity, Intent intent){
          Plugin p = mPluginMgr.loadAppPlugin(plugin);  //获取插件对象
          ......
          ActivityInfo ai = null;
      
          //activity 不为空时,从插件声明的 Activity 集合中查找
          if (!TextUtils.isEmpty(activity)) {
              ai = p.mLoader.mComponents.getActivity(activity);
          } else {
              //activity 为空时,根据 Intent 匹配
              ai = IntentMatcherHelper.getActivityInfo(mContext, plugin, intent);
          }
          return ai;
      }
      
    • PmBase.loadAppPlugin会最终调用Plugin.loadLocked()函数,这个函数有两个参数,第一个是加载类型,一共有四种加载类型,在这里使用的是Plugin.LOAD_APP,因为运行插件需要所有的东西。第二个参数是是否使用缓存,通常情况下我们会现在缓存中查找插件信息,这样会更快。只是如果大量插件加载到内存会不会占用太多的内存,感兴趣的同学可以自己研究研究。

      这里如果第一次加载失败,Replugin还会做一次重试,相关代码几乎相同,这里就省略了。

      private boolean loadLocked(int load, boolean useCache) {
          // 这里先处理一下,如果cache命中,省了后面插件提取(如释放Jar包等)操作,直接返回缓存数据
          if (useCache) {
              boolean result = loadByCache(load);
              if (result) {
                  return true;
              }
          }
          ......
          boolean rc = doLoad(logTag, context, parent, manager, load);  // 真正的加载
      
          if (rc) {
              try {
                  // 至此,该插件已开始运行
                  PluginManagerProxy.addToRunningPluginsNoThrows(mInfo.getName());
              } catch (Throwable e) {
              }
              return true;
          }
          ......
          File odex = mInfo.getDexFile();
          if (odex.exists()) {
              odex.delete();
          }
          rc = doLoad(logTag, context, parent, manager, load);
          ......
          return true;
      }
      
    • Plugin.doLoad()当然就是来加载插件的Dex文件,资源,以及so文件等等。

      private final boolean doLoad(String tag, Context context, ClassLoader parent, PluginCommImpl manager, int load) {
          if (mLoader == null) {
              // 中间省略这一段代码是释放so文件,请自行阅读代码,代码清晰简单
              ......
              // 加载Dex,获取组件信息
              mLoader = new Loader(context, mInfo.getName(), mInfo.getPath(), this);
              if (!mLoader.loadDex(parent, load)) {
                  return false;
              }
              // 在Persistent进程中更新插件信息,设置插件为“使用过的”
              try {
                  PluginManagerProxy.updateUsedIfNeeded(mInfo.getName(), true);
              } catch (RemoteException e) {
              }
      
              // 若需要加载Dex,则还同时需要初始化插件里的Entry对象
              if (load == LOAD_APP) {
                  // NOTE Entry对象是可以在任何线程中被调用到
                  if (!loadEntryLocked(manager)) {
                      return false;
                  }
              }
          }
      }
      
    • Loader.loadDex函数会获取Dex中的组件的信息,包括Manifest中的组件属性,比如进程属性,TaskAffinity属性,注册静态广播等等。但这里值得重点强调的是之前核心概念里提及的PluginDexClassLoader终于出现并被初始化了。

      final boolean loadDex(ClassLoader parent, int load) {
          try {
             ......  // 这里省略了一些基本的加载动作
              mClassLoader = Plugin.queryCachedClassLoader(mPath);
              if (mClassLoader == null) {
                  ......
                  mClassLoader = RePlugin.getConfig().getCallbacks().createPluginClassLoader(mPath, out, soDir, parent);     //  创建PluginDexClassLoader
                  ......  
              }
              ......
              mPkgContext = new PluginContext(mContext, android.R.style.Theme, mClassLoader, mPkgResources, mPluginName, this);   // 创建插件的Context对象
          } catch (Throwable e) {
              return false;
          }
      
          return true;
      }
      

      到此为止,插件Dex的加载算是全部完成,下面还剩最后一步,我们的插件就算真正的启动运行起来了。

    • 这里我们要接着前面启动坑位Activity的地方接着讲,要启动Activity首先要去加载对应的类,系统会调用ClassloaderloadClass方法,这里就是调用Replugin提供的替代者RepluginClassLoader的方法。接着又会调用PMF.loadClass,其实就是调用Pmbase.loadClass

      protected Class<?> loadClass(String className, boolean resolve) throws ClassNotFoundException {
          Class<?> c = null;
          c = PMF.loadClass(className, resolve);  
          if (c != null) {
              return c;
          }
          try {
              c = mOrig.loadClass(className); // 如果上面没有找到,那就在Host当中找
              return c;
          } catch (Throwable e) {
          }
          return super.loadClass(className, resolve);
      }
      
    • PmBase.loadClass看起来代码很多,对于Activity来说,其实我们只需要关注下面这一小段即可,mClientPluginProcessPer的实例,而PluginProcessPerIPluginClient的实现类。

      final Class<?> loadClass(String className, boolean resolve) {
          ......
          if (mContainerActivities.contains(className)) {
              Class<?> c = mClient.resolveActivityClass(className);
              if (c != null) {
                  return c;
              }
              ......
          }
          ......
      }
      
    • 这里先从PluginContainers的实例对象mACM中去查找ActivityState,对这个类还有印象吗?它就是在分配坑位的时候,我们用来保存坑位组件与真实组件对应关系的类。然后在缓存中找到插件名对应的插件对象,因为在分配坑位的时候插件信息已经加载过了,不需要重新加载。接着取出插件的ClassLoader对象,这个对象正是加载插件时创建的PuginDexClassLoader的实例了。然后利用插件的PuginDexClassLoader对象来加载真实Activity的class对象。

      final Class<?> resolveActivityClass(String container) {
          String plugin = null;
          String activity = null;
          // 找到坑位Activity与真实Activity的对应关系对象
          PluginContainers.ActivityState state = mACM.lookupByContainer(container);
          ......
          plugin = state.plugin;
          activity = state.activity;
          ......
          Plugin p = mPluginMgr.loadAppPlugin(plugin); //通过插件名从缓存中加载Plugin对象
          ......
          ClassLoader cl = p.getClassLoader();
          Class<?> c = null;
          try {
              c = cl.loadClass(activity);
          } catch (Throwable e) {
          }
          return c;
      }
      

      找到插件Activity的类对象后,Android系统就开始运行Activity的启动流程了,这些事情由ActivityManagerService和ActivityThread负责。就这样,Replugin用插件中的Activity替换了坑位Activity,我们的插件被运行起来啦!!很巧妙的设计~

小结

以上的内容就是一个插件Activity要运行起来,Replugin的基本代码流程,这里要说明一下,源码中的逻辑远不止这么点,如果你有兴趣可以跟着这篇文章在源码中过一遍,有很多不是那么复杂的逻辑这里并没有讲到,当然也有一些重要的地方因为代码并不复杂也没有讲到。

下一篇Replugin 全面解析(3) 会对插件的加载和运行做更完整和详细的讲解!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容