Android组件化开发实践

作者: wutongke | 来源:发表于2016-11-02 22:51 被阅读29642次

更新:Android组件化之通信(多模块,多进程)

Android项目中代码量达到一定程度,编译将是一件非常痛苦的事情,短则一两分钟,长则达到五六分钟。Android studio推出instant run由于各种缺陷一般情况下是被关闭的……
组件化开发可以有效降低代码模块的耦合度,使代码架构更加清晰,同时模块化的编译可以有效减少编译时间,当然总的编译时间是不会减少的,只是App模块化之后开发某个模块时,只需要编译特定模块,可以快速编译调试。

原理

组件化和插件化有些同学有些迷惑,简单来说组件化是在编译期分模块,插件化是在运行期。一般插件化用于动态修复bug或者动态更新模块,相对来说黑科技更多一些。

正常一个App中可以有多个module,但是一般只会有一个module是设置为application的,其他均设置为library,组件化开发就是要每个module都可以运行起来,因此在开发期间每个module均设置为application,发布时再进行合并。

实践

本文主要介绍一下项目组件化开发过程碰到的问题和解决办法,这里以
ModularizationApp项目为例。ModularizationApp是一个组件化的app:

Paste_Image.png Paste_Image.png Paste_Image.png Paste_Image.png Paste_Image.png
  • 架构

Paste_Image.png
其中App是主application,ModuleA和ModuleB是两个业务模块,Library是基础模块,包含所有模块需要的依赖库,以及一些工具类:如网络访问、时间工具等。代码结构如图: Paste_Image.png
  • Module作为application开发

ModuleA和ModuleB是相对独立的业务模块,可以分别进行开发,编译时只编译自身,具体实现时在gradle.properties中设置变量,如:IsBuildMudle=false
在模块的的build.gradle中:

if (IsBuildMudle.toBoolean()) {
    apply plugin: 'com.android.application'
} else {
    apply plugin: 'com.android.library'
}

在主模块的build.gradle中设置:

    if (!IsBuildMudle.toBoolean()) {
        compile project(':ModuleA')
        compile project(':ModuleB')
    } else {
        compile project(':Library')
    }

这样每个module就可以独立安装使用了,注意在修改IsBuildMudle的值时,一定要sync gradle
当module单独运行和作为module运行时,其activity在manifest中设置也会不同,这里可以根据IsBuildMudle设置不同的manifest:

    sourceSets {
        main {
            if (IsBuildMudle.toBoolean()) {
                manifest.srcFile 'src/main/debug/AndroidManifest.xml'
            } else {
                manifest.srcFile 'src/main/release/AndroidManifest.xml'
            }
        }
    }

分别在不同的目录下创建manifest文件。一定要注意两个manifest的同步问题,否则出现了莫名其名的bug,还找不到原因……

  • 资源冲突问题

当分别开发模块时,容易出资源重复命名的问题,可以在build.gradle中设置

resourcePrefix "module1_"

通过给模块设置不同的资源前缀,可以避免重复命名。

  • Activity跳转问题

拆分业务代码时,自然会涉及到跨module的Activity跳转,当单独编译时,自然是不能获取到其他模块的引用的。有几种方式可以实现跨模块的唤起Activity:

隐式启动
通过设置intent-filter实现,这需要在manifest中插入大量代码,同时也降低了安全性(其他app就可以通过这种方式随意启动)。
通过类名跳转
Android业务组件化开发实践提出了一种通过类名跳转的方式,使用脚本生成Rlist类,比较方便快捷,但感觉不方便activity间传递数据。
Scheme跳转
Scheme的方式是建立映射表,集中处理Activity,这种方式可以传递一定的数据。Activity传递大量数据时可以通过EventBus来进行传递(其实即使通过intent显示启动,也不要把大量数据放置在intent中,intent对数据大小有限制)。
在进行本次实践时找到github上的一个url Router,同时支持http和程序内Activity跳转,而且通过注解的方式进行,使用非常方便,于是引入到了项目中。项目地址ActivityRouterActivityRouter的readme中已经有比较详细的wiki,但是还有一些需要注意的:

依赖问题:

ActivityRouter使用了apt方式,因此每个使用的module中均需要设置

apt 'com.github.mzule.activityrouter:compiler:1.1.5'

注意是每个module,在Library module中设置

classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'

即可。

多module问题

ActivityRouter通过注解在编译时生成mapping,如果多个module设置依赖,就会生成多个java文件,导致源文件重复,编译出错,ActivityRouter目前提供了解决方法,但是还没有正式发布版本,可以设置:

compile 'com.github.mzule.activityrouter:activityrouter:1.1.9' 
apt 'com.github.mzule.activityrouter:compiler:1.1.6'

使用。使用在application中注解:

@Modules({"app", "moduleA", "moduleB"})
public class ModularizationApplication extends Application {
}

每个module中创建空java类注解:

@Module("moduleA")
public class ModuleA {
}

具体可以clone ModularizationApp查看代码。

编译运行

当在gradle.properties中设置IsBuildMudle=true时,可以独立运行每个module,独立运行调试,当设置IsBuildMudle=false,可以编译运行整个project,注意IsBuildMudle变量设置改变时,要对gradle进行sync

运行过程中有什么问题可以评论或者在github中提issue。

参考:
http://kymjs.com/code/2016/10/18/01
https://github.com/mzule/ActivityRouter
https://github.com/liangzhitao/ComponentizationApp

Other

欢迎关注公众号wutongke,每天推送移动开发前沿技术文章:

wutongke

推荐阅读:

Android 组件化开发原理和配置

Android组件化开发实践

Android组件化之通信(多模块,多进程)

相关文章

  • Android 组件化开发实践

    Android组件化开发实践(一):为什么要进行组件化开发?Android组件化开发实践(二):组件化架构设计An...

  • android组件化开发资料

    1、Android组件化开发实践 2、Android架构思考(模块化、多进程) 3、糯米移动组件架构演进之路 4、...

  • Android组件化

    Android 得到app 彻底组件化方案实践 CC:可关联生命周期的android组件化开发框架 美团猫眼电影a...

  • 聊聊组件化开发

    如果你在京东图书频道搜索 组件化 或者 组件化开发,显示的几乎都是 Android组件化开发 或者 Android...

  • Gradle实战——组件化的gradle build优化

    组件化gradle build优化 关于组件化,大家可以看我之前的文章,Android组件化开发实战[https:...

  • 我所理解的Android组件化之通信机制

    之前写过一篇关于Android组件化的文章, 《Android组件化框架设计与实践》 ,之前没看过的小伙伴可...

  • Android组件化和插件化开发

    Android组件化和插件化开发 什么是组件化和插件化? 组件化开发 就是将一个app分成多个模块,每个模块都是一...

  • android 收藏的一些好文

    1.常用git命令清单 2.Android组件化、模块化开发 3.Android 组件化案例 4. 5.

  • 推荐文章

    JavaJava中有关Null的9件事 AndroidAndroid组件化和插件化开发Android 组件化 ——...

  • Android组件化和插件化开发

    Android组件化和插件化开发 什么是组件化和插件化? 组件化开发就是将一个app分成多个模块,每个模块都是一个...

网友评论

  • 我是一只小壳蟆:为什么最后依赖包问题博主的demo和博客里说的是不一样的
  • Vander丶:您好,博主看了您的组件化文章之后有个疑问,在这里没有提到数据库这方面的内容,如果不同模块使用不同的数据库,这时候使用GreenDao 应该怎么处理合并的问题呢 。
  • 93a40709728c:想问下大佬,你的组件化方案,如何解决重复依赖的问题
  • qing的世界:想问一下在组件化各个module是分成不同的repo么,还是说相同的repo里面放不同的repo,谢谢
  • 放码过来吧:每个module中创建空java类注解。 这个到底在哪用上了。。。没看到哪个地方获取这个module注解并处理的地方。。
  • cheetah747:楼主楼主,为什么我组件化了之后,单独编译单个module还是慢得一逼啊?。。。。而且感觉好像有时候比编译整个项目都要慢。
  • b0acb9860013:有bug,工程有两个app,一个是全小写的app,一个是首字母大写的App. 再linux和mac环境还好,在windows下 或者 fat32 下 就会报错。
  • e27a889939f9:Arouter 组件化的 但是模块有依赖 很难拆分的
  • 张飞观察:demo导入as里面跑步起来!!!!
  • 37f5f57c12c9:你好,我也出现了RouterInit错误了,sync等方式都试过了,但是不行啊,怎么解呢?
  • acc02a490647:感谢楼主的分享,但我想问下,这种组件化的的方式使用greendao数据框架该怎么统一?我这边是通过每个model的实体类建立的表。
  • wordsMotivateme:你好,按照步骤编译下来提示 错误 找不到变量符号:RouterMapping_app.map(); 类 RouterInit
    怎么回事呢?
    wutongke:@21ea7f76b578 每次修改这个配置,相当于修改了gradle,所以需要Sync 工程,然后clean build就应该没问题啦
    21ea7f76b578:@wutongke 改动IsBuildMudle=true后,编译提示 错误 找不到变量符号:RouterMapping_moduleA RouterMapping_moduleB。是怎么回事
    wutongke:@wordsMotivateme 这个文件是根据annotation通过apt生成的,你clone下来改动过吗
  • 浩仔_Boy:你好,在下有个疑问。Fragment怎么进行组件化拆分?现在主流app基本都是一个Activity加多个Fragment这样的模式,每一个Fragment为一个模块。
    木子而东:解决了吗,同问
    叨叨宅:我们app也是这种模式,但是可以将这个之外的业务拆分出来
  • db59667c4a37:如果存在共享数据或者状态要存放在哪儿?
    db59667c4a37:@wutongke 还有,组件间通信怎么玩儿。感觉玩不好还是会耦合
    db59667c4a37:@wutongke 当然登录也可以当一个单独的组件开发。我是指,以前有不少状态和数据是放在Application中,现在有些状态和数据不再存在application,而是在各自的组件中,如何在组件间共享这样的数据?
    wutongke:@db59667c4a37 我认为可以做一个比较底层的module,比如把登录相关的功能单独作为一个module,其它module依赖它
  • 396061b58f0f:感谢楼主的分享,我在组件化的过程中碰到了个问题想请教一下:现在我将公司的App组件化之后,在抽取基础类库的过程中,有好多库是需要初始化的操作,之前是放到自定义的Application中来初始化,但是组件化之后,工程由模块组成,现在有点疑惑这些初始化的东西放到什么地方来做,请问楼主有什么好的解决方案吗?
    72c35595f71b: @396061b58f0f 搞个公共模块 里面定义application 在操作
  • 95540e8c20a4:学习了
  • 3f668a984456:真是感谢
  • EdwdChen:你好,我将IsBuildMudle改成true之后moduleB可以运行,但是moduleA运行不了,moduleA中的R没法导入到activity中,请问有办法解决吗?
    EdwdChen:@wutongke 是的,有一个manifest出问题了
    wutongke:@TheGreatEdwardW 是不是manifest文件有问题?
  • 72c35595f71b:各模块关联低情况小实用 如果关联较多 相互交叉多的就不行了
  • 1琥珀川1:有没有考虑模块过多或过大导致65536问题
    fbcde8a99e17:@1琥珀川1 设置multiDex不知道行不行。
  • af8dff7911ea:提供了一个很好的思路,但是这个的应用应该是各个模块的关联度较低的情况下使用方便。业务逻辑重的就很复杂了
    af8dff7911ea:@wutongke 是的。不过这是一个好思路
    wutongke:@无畏的coder 是的,组件化的开发,业务上的解藕反而比技术更难一些
  • 98eac8a283ce:虽然没看到十分的必要性,反而增加了复杂度,但也算—个思路
    b757ff66ad79:@wutongke 同意,几十人的工程需要这种解耦
    wutongke:@总很有神叔 嗯,如果代码量很大,编译时间过长或者有业务需求可以尝试一下。
    淡定小问题: @总很有神叔 赞同 增加了无畏的复杂度
  • arige:学习了
  • 无玄:模块与模块之前的数据传递是怎么做的呢?
    无玄:@wutongke 这几种方式都是被动的数据通知,如果模块需要拿去另外一个模块的信息,这种场景还有好的思路啊?
    全世界_gl:@dlobo 滴滴的朋友写了一个组件化通信方案。https://github.com/jackwaylong/ModuleCommunication
    wutongke:@dlobo 如果只是activity之间传递数据,可以直接使用activityrouter,如果其他的话可以使用eventbus吧,重一点多话可以使用广播
  • 我本山贼:不错的东西,值得学习!
  • Michaelhanji:嗯嗯,写的不错
  • 0b4a8a69f146:不错。在instant run还不完善的现在。配置好的话。确实能带来效率。毕竟现在公司代码已经是I7高压都要跑半分钟了
  • 口袋FPV:原来还有这么个玩法啊

本文标题:Android组件化开发实践

本文链接:https://www.haomeiwen.com/subject/wbfbuttx.html