前言
市面上有很多插件化框架:
- 林光亮的Small
- 滴滴的VirtualAPK
- 360的DroidPuglin
- 阿里的Atlas
- 百度的Dynamic-Load-APK
- 以前的旧的框架:携程、掌阅、饿了么、美团大众点评
其中很多框架已经很久没有更新维护了,我们关注点放在前面4个。
项目开发中如何抽取插件
项目中,一些关联性不强(低耦合)或者变更比较小的模块、业务就可以抽取成为一个应用插件(Bundle),例如淘宝中,搜索就是一个单独的APK。
Small的介绍
Small官网有一句话:世界那么大,组件那么小。Small,做最轻巧的跨平台插件化框架。
Small插件化方案适用于将一个APK拆分为多个公共库插件、业务模块插件的场景。其官网是https://github.com/wequick/Small,开发者是林光亮以及一些参与到Small的开发者。
下面开始介绍如何如实Small进行插件式开发。
引入Small
在项目的顶层Gradle文件引入相关的Gradle脚本:
classpath 'net.wequick.tools.build:gradle-small:1.1.0-alpha2'
然后在这个Gradle文件的末尾添加一些配置:
apply plugin: 'net.wequick.small'
small {
aarVersion = '1.1.0-alpha2'
}
主要是Small插件的应用以及版本的设定。
然后执行smsll Task检查Small是否正确引入了:
./gradlew small
引入成功的话,会输出类似的信息:
### Compile-time
```
gradle-small plugin : 1.1.0-beta4 (maven)
small aar : 1.1.0-alpha1 (maven)
gradle core : 2.14.1
android plugin : 2.2.3
OS : Mac OS X 10.12.1 (x86_64)
```
### Bundles
| type | name | PP | sdk | aapt | support | file | size |
|------|------|----|-----|--------|---------|------|------|
| host | app | | 25 | 25.0.2 | 25.1.0 | | |
配置宿主
创建一个自己的Application,调用Small.preSetUp方法:
public class SmallApp extends Application {
public SmallApp() {
Small.preSetUp(this);
}
}
创建插件module
创建一个app类型的module,注意,因为Small是根据module的名字、包名来区分宿主、插件、公共库插件的,因此起名的时候需要注意:
- 宿主module应该以app命名
- Module name 以 app.* 命名的模块将被 Small 在 编译时 识别为应用插件模块。 Package name 以 app* 结尾的插件将被 Small 在 运行时 识别为应用插件。
- Module name 以 lib.* 命名的模块将被 Small 在 编译时 识别为应用公共库插件模块。 Package name 以 lib* 结尾的插件将被 Small 在 运行时 识别为公共库插件。
下面我们以应用插件为例,创建一个应用插件:
编译插件
执行buildLib和buildBundle Task,分别编译所有公共库插件以及应用插件:
./gradlew buildLib -q
./gradlew buildBundle -q
最后,通过small Task可以看到相关的编译结果:
./gradlew small
示例如下:
type name PP sdk aapt support file(x86) size
host app 25 25.0.2 25.1.0
app app.main 0x77 25 25.0.2 25.1.0 *_appmain.so 5.3 KB
启动插件
我们在宿主里面创建一个asset文件夹,然后新建一个JSON文件:
{
"version": "1.0.0",
"bundles": [
{
"uri": "main",
"pkg": "com.example.appmain"
}
]
}
这个文件是Small插件的路由,其中:
- version,是 bundle.json 文件格式版本,目前始终为 1.0.0
- bundles,插件数组:uri,插件唯一ID;pkg,插件包名
最后,通过宿主调用下面这段代码即可路由到应用插件:
Small.setUp(this, new Small.OnCompleteListener() {
@Override
public void onComplete() {
Small.openUri("main", MainActivity.this);
}
});
注意事项
每次运行、打包的时候,如果插件有更新,最好完整走一遍下面的流程:
- 执行./gradlew buildLib -q编译公共库插件
- 执行./gradlew buildBundle -q编译应用插件
- 执行./gradlew small检查编译是否正确生效
- run或者apk打包
Small公共库
网络请求、公共资源等可以抽取成为一个单独的module,然后进一个变成公共库插件。
但是公共库插件的使用会有一些坑,这里笔者不建议大家使用,还是老老实实添加module之间的依赖。
Small使用进阶
Small的基本使用就介绍到这里了,下面还有一些进阶的话题:
- 自定义资源ID分段:在整合插件资源的过程,为避免资源ID冲突,需要为每个插件分配一个ID段。
- 插件路由、编译选项仔细讲解。
- DSL规范等
在实际使用到Small的时候,以上都可以直接参考官方资料:http://code.wequick.net/Small/cn/home,这里不再赘述。