在Android开发领域,有关插件化的讨论一直热度不减。目前市面上的插件化方案虽然很多,但多数只能实现某些功能的插件化,距离开发者的预期尚有相当差距。对此,在近期GMTC全球移动技术大会上,360手机卫士主程序架构负责人张炅轩宣布,360的插件化框架RePlugin已经可以实现“全面插件化”,同时具有出色的稳定性和灵活性,可适用于各种类型的应用上。
“RePlugin预计7月份开源,这将是我们献给安卓世界最好的礼物。”360如是说。
RePlugin是一套完整的、稳定的、适合全面使用的,占坑类插件化方案,由360手机卫士的RePlugin Team研发,也是业内首个提出”全面插件化“(全面特性、全面兼容、全面使用)的方案。
其主要优势有:
* 极其灵活:主程序无需升级(无需在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年验证 |
特性 | 描述 |
---|---|
组件 | 四大组件(含静态Receiver) |
升级无需改主程序Manifest | 完美支持 |
Android特性 | 支持近乎所有(包括SO库等) |
TaskAffinity & 多进程 | 支持(坑位方案) |
插件类型 | 支持自带插件(自识别)、外置插件 |
插件间耦合 | 支持Binder、Class Loader、资源等 |
进程间通讯 | 支持同步、异步、Binder、广播等 |
自定义Theme & AppComat | 支持 |
DataBinding | 支持 |
安全校验 | 支持 |
资源方案 | 独立资源 + Context传递(相对稳定) |
Android 版本 | API Level 9+ (2.3及以上) |
ok,看完了官方介绍给我们画的大饼,现在看看怎么实现它,是的,坑来了。。
主程序的配置,这里就不多说了。。将RePlugin接入到您的主程序,官方文档描述的很清楚,也没有什么奇奇怪怪的错误。
主要来说说如何开发新的 RePlugin 插件,从RePlugin的Wiki我们可以看到,好简单呀,就三步嘛,来吧。
然后我们碰到了………
EXCUSE ME?那里有问题?再看看文档,没啥特别的描述呀。
然后,我在Issues里找到了官方项目组说的这一句:“这个要自动读取前面build.gradle的配置内容,如果放在前面,会读为空。我们内部讨论过这个问题,如果要想自由放置就得传参配置。当时考虑到尽量减少传参配置就约定俗成让放在文件末尾。你可以看demo1。”
好吧,既然如此能不能提一句只能放在文件末尾啊!
你是360,你老大,我改!
??按你说的还不行??
然后,我在Issues里看到了吃瓜群众说的这一句:“你试试把apply plugin: 'replugin-plugin-gradle'
放在apply plugin: 'com.android.application'
之前就好了,我的就是这样好使的,你试试?”
抱着试一试的心态,然后我发现…成功了??说好的读取配置内容呢?说好的会读为空呢?
OK,按照官方文档,各项都配置完成,我们继续进行。。
插件界面很简单:
插件单独运行一下,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" 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的功能性,肯定是毋庸置疑的,应用场景也必将十分广泛。相信在未来很多项目中,会见到它的身影。