×

Android全面插件化方案-RePlugin踩坑

96
Sakura_echo
2017.07.04 16:08* 字数 1452

1.什么是RePlugin?

在Android开发领域,有关插件化的讨论一直热度不减。目前市面上的插件化方案虽然很多,但多数只能实现某些功能的插件化,距离开发者的预期尚有相当差距。对此,在近期GMTC全球移动技术大会上,360手机卫士主程序架构负责人张炅轩宣布,360的插件化框架RePlugin已经可以实现“全面插件化”,同时具有出色的稳定性和灵活性,可适用于各种类型的应用上。
“RePlugin预计7月份开源,这将是我们献给安卓世界最好的礼物。”360如是说。

2.RePlugin有什么用?

RePlugin是一套完整的、稳定的、适合全面使用的,占坑类插件化方案,由360手机卫士的RePlugin Team研发,也是业内首个提出”全面插件化“(全面特性、全面兼容、全面使用)的方案。

3.RePlugin官方介绍

其主要优势有:

  • 极其灵活:主程序无需升级(无需在Manifest中预埋组件),即可支持新增的四大组件,甚至全新的插件
  • 非常稳定:Hook点仅有一处(ClassLoader),无任何Binder Hook!如此可做到其崩溃率仅为“万分之一”,并完美兼容市面上近乎所有的Android ROM
  • 特性丰富:支持近乎所有在“单品”开发时的特性。包括静态Receiver、Task-Affinity坑位、自定义Theme、进程坑位、AppCompat、DataBinding等
  • 易于集成:无论插件还是主程序,只需“数行”就能完成接入
  • 管理成熟:拥有成熟稳定的“插件管理方案”,支持插件安装、升级、卸载、版本管理,甚至包括进程通讯、协议版本、安全校验等
  • 数亿支撑:有360手机卫士庞大的数亿用户做支撑,三年多的残酷验证,确保App用到的方案是最稳定、最适合使用的

截止2017年6月底,RePlugin的:

特性 描述
插件数 103(核心57个)
插件占应用比 高达83%
年发版次数 高达596次(工作日均2次)
崩溃率 万分之一(0.01%),极低
时间 2014年应用,3年验证

目前360公司几乎所有的亿级用户量的APP,以及多款主流第三方APP,都采用了RePlugin方案。

有关RePlugin的详细介绍,请点击这里阅读《RePlugin 官方 WiKi》

还支持以下特性

特性 描述
组件 四大组件(含静态Receiver)
升级无需改主程序Manifest 完美支持
Android特性 支持近乎所有(包括SO库等)
TaskAffinity & 多进程 支持(坑位方案)**
插件类型 支持自带插件(自识别)、外置插件**
插件间耦合 支持Binder、Class Loader、资源等
进程间通讯 支持同步、异步、Binder、广播等
自定义Theme & AppComat 支持
DataBinding 支持
安全校验 支持
资源方案 独立资源 + Context传递(相对稳定)
Android 版本 API Level 9+ (2.3及以上)

愿景

让插件化能飞入寻常应用家,做到稳定、灵活、自由,大小项目兼用。

RePlugin 架构图

RePlugin Framework
RePlugin Framework

4.SO?怎么用?

ok,看完了官方介绍给我们画的大饼,现在看看怎么实现它,是的,坑来了。。
主程序的配置,这里就不多说了。。将RePlugin接入到您的主程序,官方文档描述的很清楚,也没有什么奇奇怪怪的错误。
主要来说说如何开发新的 RePlugin 插件,从RePlugin的Wiki我们可以看到,好简单呀,就三步嘛,来吧。
然后我们碰到了.........

问题一:Gradle配置出错

gradle配置出错.png

EXCUSE ME?那里有问题?再看看文档,没啥特别的描述呀。
然后,我在Issues里找到了官方项目组说的这一句:“这个要自动读取前面build.gradle的配置内容,如果放在前面,会读为空。我们内部讨论过这个问题,如果要想自由放置就得传参配置。当时考虑到尽量减少传参配置就约定俗成让放在文件末尾。你可以看demo1。”
好吧,既然如此能不能提一句只能放在文件末尾啊!
你是360,你老大,我改!

gradle配置出错.png

??按你说的还不行??
然后,我在Issues里看到了吃瓜群众说的这一句:“你试试把apply plugin: 'replugin-plugin-gradle'放在<apply plugin: 'com.android.application'之前就好了,我的就是这样好使的,你试试?”
抱着试一试的心态,然后我发现...成功了??说好的读取配置内容呢?说好的会读为空呢?

OK,按照官方文档,各项都配置完成,我们继续进行。。
插件界面很简单:


image.png

插件单独运行一下,ok,没问题。
好的,接下来走一下内置插件的流程,将插件项目build一个apk出来,后缀改成jar,导入主程序assets的plugins内,
主程序界面也很简单,就一个HelloRePlugin的TextView,给TextView设置一个点击事件

tv.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                RePlugin.startActivity(MainActivity.this, RePlugin.createIntent("aap2", "com.zzcn77.replugindemo2.MainActivity"));
            }
        });

然后我们碰到了.........

问题二:主题出错

点击HelloRePlugin,程序崩溃。

主题出错

主题错了?继承的是Activity啊,有主题啊,单独运行没有错啊。。
自此,我开始多番尝试之路,改了n个主题发现没有用,气的我,直接把apply plugin: 'replugin-plugin-gradle'
我把这句去掉了,运行。。程序调起成功了。。。我的天哪
诡异的一幕出现了,吊起的插件Acitivity显示界面:
主题错误

???这个界面哪来的?,这不是我的插件界面啊。。
奇怪的是,插件的activity的onCreate也走了。那我的界面去哪了?
是不是因为我删去了apply plugin: 'replugin-plugin-gradle',所以出现问题了?,还是加上吧,再看看主题出错有没有其他解决方法。------加上,buildApk,导入主程序,运行主程序,点击HelloRePlugin,调起,成功了?!,唉?你不是主题有问题的吗?你不是主题有问题的吗?你不是主题有问题的吗?好吧,,虽然调起成功,可是还没完,打开的activity界面依然如上。

网络图片

然后我们发现了.........

问题三: Didn't find class "com.qihoo360.plugin.app2.Entry"

 Didn't find class "com.qihoo360.plugin.app2.Entry" on path: DexPathList[[zip file "/data/user/0/com.replugindemo/app_plugins_v3/app2-10-10-101.jar"],nativeLibraryDirectories=[/data/user/0/com.replugindemo/app_plugins_v3_libs/app2-10-10-101, /vendor/lib, /system/lib]]
 java.lang.ClassNotFoundException: Didn't find class "com.qihoo360.plugin.app2.Entry" on path: DexPathList[[zip file "/data/user/0/com.replugindemo/app_plugins_v3/app2-10-10-101.jar"],nativeLibraryDirectories=[/data/user/0/com.replugindemo/app_plugins_v3_libs/app2-10-10-101, /vendor/lib, /system/lib]]
                 at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:56)
                 at java.lang.ClassLoader.loadClass(ClassLoader.java:511)
                 at com.qihoo360.replugin.PluginDexClassLoader.loadClass(PluginDexClassLoader.java:76)
                 at java.lang.ClassLoader.loadClass(ClassLoader.java:469)
                 at com.qihoo360.loader2.Loader.loadEntryMethod2(Loader.java:419)
                 at com.qihoo360.loader2.Plugin.loadEntryLocked(Plugin.java:857)
                 at com.qihoo360.loader2.Plugin.doLoad(Plugin.java:822)
                 at com.qihoo360.loader2.Plugin.loadLocked(Plugin.java:621)
                 at com.qihoo360.loader2.Plugin.load(Plugin.java:432)
                 at com.qihoo360.loader2.PmBase.loadPlugin(PmBase.java:1033)
                 at com.qihoo360.loader2.PmBase.loadAppPlugin(PmBase.java:1018)
                 at com.qihoo360.loader2.PmLocalImpl.getActivityInfo(PmLocalImpl.java:443)
                 at com.qihoo360.loader2.PmLocalImpl.loadPluginActivity(PmLocalImpl.java:319)
                 at com.qihoo360.loader2.PmInternalImpl.startActivity(PmInternalImpl.java:230)
                 at com.qihoo360.loader2.PmLocalImpl.startActivity(PmLocalImpl.java:307)
                 at com.qihoo360.i.Factory.startActivityWithNoInjectCN(Factory.java:324)
                 at com.qihoo360.replugin.RePlugin.startActivity(RePlugin.java:236)
                 at com.zzcn77.replugindemo.MainActivity$1.onClick(MainActivity.java:24)
                 at android.view.View.performClick(View.java:5198)
                 at android.view.View$PerformClick.run(View.java:21147)
                 at android.os.Handler.handleCallback(Handler.java:739)
                 at android.os.Handler.dispatchMessage(Handler.java:95)
                 at android.os.Looper.loop(Looper.java:148)
                 at android.app.ActivityThread.main(ActivityThread.java:5417)
                 at java.lang.reflect.Method.invoke(Native Method)
                 at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
                 at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
                 Suppressed: java.lang.ClassNotFoundException: com.qihoo360.plugin.app2.Entry
                 at java.lang.Class.classForName(Native Method)
                 at java.lang.BootClassLoader.findClass(ClassLoader.java:781)
                 at java.lang.BootClassLoader.loadClass(ClassLoader.java:841)
                 at java.lang.ClassLoader.loadClass(ClassLoader.java:504)
                     ... 25 more
Caused by: java.lang.NoClassDefFoundError: Class not found using the boot class loader; no stack trace available

为什么?
再看看官方文档,遗漏什么了?没有啊。
再看看官方文档,还有没看到的?没有啊。
为什么?
再回想一下整个流程,gradle出错,主题样式错误,Didn't find class "com.qihoo360.plugin.app2.Entry
对,主题样式,这一环并没有真正的解决。
找不到主题?application里面配置的有主题呀。
把application的统一主题去掉,分别给每个activity配置主题。
再来一次,buildApk,导入主程序,运行主程序,调起,成功了!!!完美运行!!!
掌声,掌声,掌声

总结

首次的整个集成过程,也算是踩坑无数,乐趣多多啊。希望Replugin项目组在后续的版本中能够提供更多的功能,以及更高的稳定性,当然也希望官方的Wiki再友好一点,能对一些集成规范的描述更详细点呗。不过好用是真的。
作为行业大哥360开源出来的全面化插件机制,以及在360众多项目中的实践,Replugin的功能性,肯定是毋庸置疑的,应用场景也必将十分广泛。相信在未来很多项目中,会见到它的身影。

网络图片
Android随笔
Web note ad 1