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

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的功能性,肯定是毋庸置疑的,应用场景也必将十分广泛。相信在未来很多项目中,会见到它的身影。

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

推荐阅读更多精彩内容