Android 组件化架构(一)

组件化就是多module划分业务和基础功能。这样可解耦,提高代码复用。在正式开始讲解组件化之前可以先看一下一个组件化架构图来对组件化有一个初步的认识。


架构图

1.基础层中一般会包括一些基础库,比如图片加载,网络加载,数据存储等等。
2.组件层就是业务逻辑相关的代码,比如登录、支付等
3.应用层就是把所有的组件组合起来,发布一个完整的App

组件层中有很多Activity,这些Activity都需要在相应Module中AndroidManifest中注册,合并成一个App的过程中,Manifest也会被合并,这正是组件化能够进行的原因。

1.组件化Application

和每个Module中都可以有Activity一样,每个Module中也可以有一个Application,如果不做任何处理,直接添加一个Module,编译之后就会报错Manifest Merge failed:Attribute application @name value=(xxx.xx.xApplication) from AndroidManifest.xml is also present at xxxx...这是因为AndroidStudio的Gradle插件默认会启用Manifest Merge Tool,如果Module 项目中也定义了与主项目相同的属性,如(android:icon或者android:theme),就会合并失败。

可以在Module中Manifest文件中添加tools:replace="android:name,android:theme"来声明Module的Application中相关属性是可以被替代的,这样就可以解决冲突。

通常我们需要做一些全局弹窗,比如常见的App升级弹窗,或者一些广告弹窗,那么我们就需要获取顶层Activity的Context并传递给需要弹出的Dialog。如何获取顶层的Activity?这时就需要介绍一个方法:Application的registerActivityLifecycleCallback(),这个方法可以监听App所有的Activity活动,包括各种生命周期事件。当然也能获取顶层Activity对象。

2.组件化编程

谈到组件化编程前我们先思考一个问题:如何启动一个无法获取类名的Activity?一个简单的方法就是隐式意图。关于如何使用隐式意图不在本文的讨论范围内,不熟悉的童鞋请自行查看相关代码。

组件化编程为什么要涉及到这个问题呢?因为组件化编程中,不同的页面会在不同的Module中,Module和Module之间是相互隔离(再看一遍架构图),无法直接访问到对方。既然无法访问到对方,肯定无法得到需要启动的Activity的类名的。这就延伸出了上一段提到的问题。当然是用隐式意图可以做到这一点,但是除了不安全外(整个系统都可以监听到这个隐式意图),更多是考虑到编写隐式意图代码的繁杂,有大量的boring code。有没有更好的替代方案?

这就引申出了一些组件化路由方案的存在。

什么是路由?通俗的说即从一个地方到另一个地方。这个词经常出现在计算机网络术语中,表示路由器从端口获取数据并根据数据包的内容分发到指定端口的过程。

我们的Activity就像一个一个端口,跳转的Intent就像数据,我们需要找到一种合适方法将intent分发到指定的Activity。很多第三方框架已经可以很好地帮我们完成这个过程,比如ARouter。在路由的过程中可以对路由的信息做一些处理,当然这就和业务逻辑向关联起来了,以后再做讨论。

首先需要在Base module中添加一些配置,导入ARouter库。详细的配置过程参考ARouter的接入Wiki。
接入完成后我们跳转Activity的代码就变成了:

ARouter.getInstance().build("/login/register")
              .withString("name","laotie")
              .navigation();

这里面build("/login/register")是指想要跳转的Activity的@Route("/login/register")注解名称,withString("name","laotie")相当于需要传递到Intent中的param。跳转目标的Activity可以像获取普通Intent中的参数一样获取里面的值。"/login/register"中login就是groupName,不同的Module应该有不同的groupName。否则会编译报错。

ARouter后面的原理是在Application加载ARouter的init()方法后,建立起"login/register"和相应Activity的映射关系,如果查看源码,跳转的时候本质上还是调用的startActivty()

3.反射和组件化编程

通常我们采用的是Activity+Fragment的方式来将页面和Activity解耦。在组件化架构就会出现在这样一个问题,这里需要的Fragment在另个一Module里面。直接引用相应的Fragment除了Module之间不应该直接访问外,如果移除了相应的Module,必然会带来找不到类的错误。这时我们就需要通过反射的方式到相应的module中获取该Fragment的实例。(虽然反射效率降低,但是可以提高解耦)反射获取相关的类的知识点请自行研究。

和隐式意图一样,手动去写反射的代码也很繁杂。ARouter同样提供了很好地跨模块解决方案:Fragment f = (Fragment) ARouter.getInstance().build("/login/register_fragment").navigation();拿到了Fragment就可以加载到Activity中,如果查看源码的话,底层的原理依然是反射。

可是每次都这么写的话三个module就需要把相同的代码写三遍,要是能在Fragment所在的Module中有一个提供该Fragment的方法,其他模块只需跨模块调用这个方法即可添加相应的Fragment。在这样的想法下想把Fragment初始化的方法解耦到每个需要的module中,就需要跨module调用方法。ARouter也提供了跨模块调用对象的方法。

  1. 扩展IProvider接口,创建一个初始化参数接口
public interface FragmentProvider extends IProvider{
        Fragment newInstance(Activity activity,@IdRes int containerId,FragmentManager manager,Bundle bundle,String tag);
}
  1. 继承扩展接口并实现初始化Fragment的方法
@Router(path="/login/register_fragment_provider")
public class NewFragmentProvider implements FragmentProvider{
        @Override
        public void init(Context context){
        }

        @Override
        public Fragment newInstance(Activity activity,@IdRes int containerId,FragmentManager manager,Bundle bundle,String tag){
              return ViewUtils.(activity, containerId, manager, bundle,RegisterFragment.class, tag);
        }
}
  1. 通过ARouter的IProvider接口和路由地址调用初始化Fragment方法。
((FragmentProvider) ARouter.getInstance().build("/login/register_fragment_provider")
.navigation()).newInstance(activity, containerId, manager, bundle, tag);

这样我们就可以在不同的module中调用方法获取Fragment,也避免了很多重复的代码。

谈完了Activity和Fragment,让我们再回过头来看看Application。

通常我们会在Application中初始化一些第三方库。但是一些第三方库只和对应的module有关系,如果强制把他们塞到base module的gradle中就会显得很臃肿,而且这个Module如果被移除了,我们用不到这个第三方库,但是我们的工程中依然会有这个第三方库,这样就无形增大工程的无用代码和app的体积。如何解决这个问题?

到对应的Module中引用并初始化这些类库!说的简单,Module中没有Application,怎么做到这一点?

1.方法一 反射

在Moduel中定一个初始化类Init.class,里面的init()方法会初始化相应的库。和直接获取Fragment一样,我们可以定义好该初始化类在工程中的路径,通过反射获取该类,然后调用init()方法即可。

2.方法二 通过ARouter的跨Module方法调用完成初始化

有没有感觉很熟悉,既然第一个方法和直接获取Fragment类似,那么对应的,我们也可以像第二种添加Fragment的方法一样,通过IProvider类,直接提供对初始化类Init.class的init()方法的调用。详细的代码参考上面的就可以了。

到这里就可以粗略的对组件化有一个大概的认识,知道了怎么去构建一个简单的组件化框架。下面一篇文章将会介绍如何在不同的Module中通信、存储数据、管理权限请求等,还有如何更好地解耦等等。慢慢一步一步填充组件化开发过程中的细节问题。

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

推荐阅读更多精彩内容