谈谈我理解的Android组件化

Android组件化分享

为什么要做组件化

一个App总归是要迭代更新的,这个过程中,业务逻辑也会慢慢增加或者修改的越来越复杂,这样业务模块也就是对应的package继续增加是不可避免的
,相应的每个模块的代码只会变多,所以单一工程下的APP或者说单一业务组件的架构极有可能会影响开发效率
,站在新员工的角度来看,每个伙伴着手前需要熟悉如此多的代码,较难上手,而且编译代码时间会非常卡,开发过程中,出现问题需要跑整个项目,所以必须要有更灵活的架构代替过去单一的工程架构。

认识一下组件化

先来解释一下组件化两种模式

  • 集成模式:所有的业务组件(module)都是被空壳(app module)依赖,合成一个完整的项目.
  • 组件模式:可以单独运行编译出独立的项目,简单的说就是一个组件一个app

再来看看切割的业务组件和功能组件

  • app module:原本单一工程的主角,大部分的业务都写在其中,甚至功能工具,现在他是一个空壳,用来整合各个业务组件(a module……),负责打包apk等,没有具体的业务功能
  • launch module 也算半个业务组件,负责制定APP启动界面。
  • a module 根据a业务组件独立形成的一个工程
  • b module 根据b业务组件独立形成的一个工程
  • c module 根据c业务组件独立形成的一个工程
  • common module 一个功能组件,为业务组件提供对应的功能(可细拆分功能)

其实已经很清晰了,简单一点说就是,组件化就是将从前的模块化的东西,拆成了组件形式,common组件问题不大,一般app架构里都会有这么一个功能组件,组件模式后单独运行代码量,少之又少,
可以提高速度,方便测试。
这里有一点是需要考虑的,就是并不是所有模块都是适合拆出来成为组件,成为一个特立独行的工程,拆成组件需要对业务有比较深的理解,哪些业务是紧密连接的,哪些业务是可切割的。
不是组件越多越好,而应该以组件切割得清晰来衡量这个架构的水平。
我的理解是,其实在上面已经说到过,工程这个词,如果拆出来的模块能构成一个小工程来运行,或者说可以帮助项目解耦,方便单元测试,甚至是编译速度,那么它都是可拆的。

组件化流程与问题

组件模式与集成模式的切换
apply plugin: ‘com.android.application’ 对应的是Android应用程序,也就是我们的App,可以独立运行
apply plugin: ‘com.android.library’ 对应的是Android 库文件,可以理解为本地库,不可独立运行

每个组件的属性都放在build.gradle文件中,其中控制这两个模式的属性,一般就在文件第一行。
业务组件处在application属性时,这个组件就是一个工程,独立运行,开发和调试,当处在library时,他才可以被app空壳工程依赖,与其他业务组件合成一个完整的app。
那么要如何切换这个属性呢?
肯定是不能每次都修改build.gradle文件的属性的,必须需要一个开关来决定这个组件的模式,这时候就需要一个常量来判断,我所知道的有两种方式创建这个常量。

1、其实在项目根目录下有一个**gradle.properties**文件,在Android项目中的任何一个**build.gradle**文件中都可以把**gradle.properties**中的常量读取出来。
2、或者你定义一个全局配置**config.gradle**,在系统级别的**build.gradle**把**config.gradle**apply进去,在**config.gradle**文件中定义常量

定义一个常量值isAModuleApplication*(true为集成模式,false为组件模式),操作如下:

需要注意的是,取出来的值,它是String类型,这时候需要以下写法
if (isAModuleApplication.toBoolean()) {
    apply plugin: 'com.android.application'
} else {
    apply plugin: 'com.android.library'
}

改完之后,同步一下就可以看到效果了

AndroidManifest清单文件合并问题
  • 一个组件当它是组件模式的时候,他的AndroidManifest需要几个作为application应用(也就是App工程)的东西,特别是声明一个application和设置一个入口(启动界面)。
  • 一个组件当它是集成模式的时候,它的AndroidManifest会被合并到app空壳工程里,那么一个工程不应该要有多个入口或者多个application。。

那么问题来了,怎么才能让它是组件模式的时候有对应的东西,集成的时候又抹除不该有的?

答案很简单,需要有两个AndroidManifest清单文件,一份作为组件模式独立运行使用,一份作为集成模式被app空壳依赖使用,还要两份对应的各自的application对象。

现在就是要让程序知道在不同模式下使用不同的AndroidManifest清单文件和application。

在main文件夹下面创建一个runalong文件夹,new一个清单文件,文件夹名字可以随便取,意思要到位,独立运行!
在java文件夹下面创建一个runalong文件夹,new一个自定义的application对象,文件名字可以随便取。

这时候有2个清单文件和application,需要程序自己取了,在业务组件下的**build.gradle**中指定清单文件的路径,操作如下
 sourceSets {
        main {
            if (isAModuleApplication.toBoolean()) {
                manifest.srcFile 'src/main/AndroidManifest.xml'
            } else {
                manifest.srcFile 'src/main/runalong/AndroidManifest.xml'
                java {
                      exclude 'runalong/**'
                }
            }
        }
    }
再来看看2个清单文件的内容:
    组件模式
    <application
          android:name="runalong.XxxApp"
          android:icon="@mipmap/ic_launcher"
          android:label="@string/app_name"
          android:persistent="true"
          android:supportsRtl="true"
          android:theme="@style/AppTheme"
          tools:replace="android:label"
          tools:ignore="GoogleAppIndexingWarning">
          <activity
              android:name=".main.MainActivity"
              android:screenOrientation="landscape"
              android:windowSoftInputMode="stateAlwaysHidden">
              <intent-filter>
                  <action android:name="android.intent.action.MAIN"/>
  
                  <category android:name="android.intent.category.LAUNCHER"/>
              </intent-filter>
          </activity>
  
          ....
    </application>
    
    集成模式
    <application android:theme="@style/AppTheme">
           <activity
                android:name=".main.MainActivity"
                android:screenOrientation="landscape"
                android:windowSoftInputMode="stateAlwaysHidden">
                
                
           </activity>
    
           ....
     </application>
     
可以看到,组件模式的时候,一个app需要的东西一个都不能少,集成模式的时候,基本上是一个都不能要。

因为处在组件模式,不需要空壳做任何操作,那么可以如下操作

if(isAModuleApplication.toBoolean()){
    java {
          exclude 'com/xxx/xxx/**'
    }
}
全局Context的获取

开发过程中,一般我们会自定义一个继承Application的对象,来获取全局Context。
现在要做的是,不管处在什么模式下都能获得全局Context
上面提到过,当我们在组件模式开发中,每一个组件都要有application,所以我们在java文件夹下面创建一个runalong文件夹,同时声明一个application来支持组件特立独行。。一切看似都很美好
当我们切换到集成模式的时候,会发现runalong中的application没有执行,因为main文件夹下runalong下的清单文件被排除了,所以只有app空壳工程中的application才有全局Context。
现在我们就需要用到common module(公用功能组件)了,定义一个BaseApplication,继承Application,因为app空壳工程依赖common组件,所以将app空壳工程中的自定义的application
对象继承BaseApplication,并且,在app空壳工程中的清单文件中声明这个自定义的application对象,以确保集成模式启动时,common组件中的BaseApplicaition被执行,至此,保证集成模式下
其他业务组件都可以获取的到全局的Context对象。
需要注意的是,其他业务组件在独立运行的时候,需要将runalong文件夹下的自定义application对象继承common组件中的BaseApplication,并在其runalong文件夹下的清单文件中声明,保证组件模式下
的common组件中的BaseApplication被执行。
所以不管是组件模式独立运行还是集成模式都可以获取全局Context对象。

lib第三方库的依赖

项目中多少都会使用到一些实用的库,当多人协作开发时,每个人基本上是管好自己的项目,这样会造成第三方库重复甚至泛滥。
所以

  • 首先需要对第三方库进行评估,尽量排除不稳定或者不更新的lib
  • 为了统一管理,我们将第三方库放在common组件中,提供给业务组件
  • 在common组件中,我们需要使用api(这里效果是和compile是一样的),不能使用implementation来加载,implementation只会在自身组件中使用,不能对外提供。
组件之间的通信

因为组件之间没有相互依赖,所以不存在直接调用,那么需要如何调用呢??
首先想一下,我们每个组件都有依赖一个叫做common的组件,我们依然还是需要它作为中间的一个桥梁,帮助我们让海峡两岸进行沟通,开始做桥梁吧

  • 我们需要一个桥梁管理器,BridgeManager,用来管理无数个桥梁,为每个actitvity制定一个易于管理的名字,用功能/包名+类名,如vip/com.xxx.xxx.VipActivity,来命名。
  • BridgeManager注册这些名字,存在Map<String,Class>中,以便提取。
  • 提取过程中,将制定的名字切割,用反射获取到指定包下的activity,就可以进行组件通信了。
public static final String VIP_VIP = "vip/com.xxx.xxx.VipActivity";

public class BridgeManager {

    private static final String TAG = "BridgeManager";

    private static HashMap<String, Class<Activity>> hashMap = new HashMap<>();

    public static Class<Activity> findBridgeObj(String bizName) {
        String className = parseBizName(bizName);
        if (TextUtils.isEmpty(className)) {
            return null;
        }
        Class<Activity> bridgeObject = hashMap.get(className);
        if (bridgeObject == null) {
            bridgeObject = createBridgeObject(className);
        }
        return bridgeObject;
    }

    private static boolean register(Class<Activity> activityClass) {
        if (activityClass == null) {
            return false;
        }

        String classNameKey = activityClass.getName();
        if (hashMap.containsKey(classNameKey)) {
            Log.e(TAG, "请勿重复注册 key" + classNameKey);
        }
        hashMap.put(classNameKey, activityClass);
        return true;
    }

    private static Class<Activity> createBridgeObject(String className) {
        if (TextUtils.isEmpty(className)) {
            return null;
        }

        //反射
        Class<Activity> activityClass = null;
        try {
            Class<Activity> clazz = (Class<Activity>) Class.forName(className);
            if (register(clazz)) {
                activityClass = clazz;
            }
        } catch (Exception e) {
            Log.e(TAG, e.getMessage());
        }
        return activityClass;
    }

    private static String parseBizName(String bizName) {
        if (TextUtils.isEmpty(bizName)) {
            return null;
        }
        int index = bizName.indexOf("/");
        if (index != -1) {
            return bizName.substring(index + 1);
        } else {
            throw new IllegalArgumentException("not found the bizName :" + bizName);
        }
    }
}

public static void startAct(Context context, String bizName) {
        Class<Activity> activityClass = BridgeManager.findBridgeObj(bizName);
        context.startActivity(new Intent(context, activityClass));
    }

过程很简单,就是利用反射获取包名进行调用,怎么封装也有很多花样,这里只是提供一个思路,还是极力推荐使用ARouter进行组件通信,方便快捷,可以了解一下。

资源文件命名问题与规范

单多个协同开发时,难免存在一些资源文件上的命名冲突,比如都有一个drawable_background的drawable文件,两个命名如果是一样的,在集成模式下会导致编译不通过。
最直接的办法就是组内人员规定某些命名,但是不可估计和预判的资源文件是没法说明哪个文件用哪个命名,所以只能在资源文件名的头部,加上我们的组件名,如,a_drawable_background,b_drawable_background
这里还存在一个问题,因为人做事总会疏忽,不是这次就是下次,所以有没有办法约束一下命名,答案是有!

android{
    ......
    
    resourcePrefix vip_
    
    .....
}

这样每次创建新的资源文件,都会强制要求你文件名必须以vip_开始,否则就会报红,虽然并不影响编译和运行,但是会有一个强烈的错误警告,起到很好的提示作用
值得一提的是图片也是属于资源文件,但是并不会对图片命名有约束,这个一点还是要开发人员手动修改,或者根据使用场景规范命名。
BuildConfig.DEBUG始终为true

开发中一般会通过 BuildConfig.DEBUG 判断是否是 Debug 模式,从而做一些在 Debug 模式才开启的特殊操作,比如打印日志。这样好处是不用在发布前去主动修改,因为这个值在 Debug 模式下为 true,Release 模式下为 false。
如果应用只有一个 Module 没有问题,Debug 模式下BuildConfig.DEBUG 会始终为 true。如果现在有两个Module,会有问题。
比如一个A module和common module,common module中的日志工具中使用了BuildConfig.DEBUG来判断是否输出日志,那么永远都是false。
BuildConfig.java 是编译时自动生成的,并且每个Module都会生成一份,所以如果你的应用有多个 Module 就会有多个 BuildConfig.java 生成。
而上面的common module import 的是自己的BuildConfig.java,编译时被依赖的 Module 默认会提供 Release 版给其他 Module 或工程使用,这就导致该 BuildConfig.DEBUG 会始终为 false。
解决方案,我有两种:

  • 始终调用最终运行的Module的BuildConfig,因为它没有被任何其他Module依赖,所以BuildConfig.DEBUG 值会准确。
public class AppUtils {
 
    private static Boolean isDebug = null;
 
    public static boolean isDebug() {
        return isDebug == null ? false : isDebug.booleanValue();
    }
 
    /**
     * Sync lib debug with app's debug value. Should be called in module Application
     *
     * @param context
     */
    public static void syncIsDebug(Context context) {
        if (isDebug == null) {
            isDebug = context.getApplicationInfo() != null &&
                    (context.getApplicationInfo().flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0;
        }
    }
}
  • 让被依赖的Module提供除Release版以外的其他版本
android {
    publishNonDefault true
}
表示该Module打包时会同时打包其他版本,包括Debug版。并且需要在App空壳中将其依赖的common如下逐个添加:
dependencies {
    releaseImplementation project(path: ':common', configuration: 'release')
    debugImplementation project(path: ':common', configuration: 'debug')
}
表示依赖不同版本的common Module。
组件化三种工程类型的build.gralde
  • app空壳工程
  • common功能组件
  • 业务组件

app空壳工程

与单一工程的**build.gradle**并没有什么不同,需要注意的是根据isModuleApplication来选择引入不同的依赖,和排除不同模式下不需要的文件夹,以下是一份app空壳工程的简单build.gradle
apply plugin: 'com.android.application'

android {
    compileSdkVersion rootProject.ext.android.compileSdkVersion

    defaultConfig {
        applicationId rootProject.ext.android.applicationId
        minSdkVersion rootProject.ext.android.minSdkVersion
        targetSdkVersion rootProject.ext.android.targetSdkVersion
        versionCode rootProject.ext.android.versionCode
        versionName rootProject.ext.android.versionName
        testInstrumentationRunner rootProject.ext.android.AndroidJUnitRunner
        multiDexEnabled rootProject.ext.android.multiDexEnabled
        
        ....
    }
    
    ....

    sourceSets {
        main {
            if (isAModuleAppliction.toBoolean()) {
                java {
                    exclude 'com/xxx/xxx/**'
                }
            }
            
            ....
        }
    }

    ....
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])

    implementation project(':common')
    if (!isAModuleCashAppliction.toBoolean()) {
        implementation project(':a_module')
    }
    
    ....
}

common功能组件

不管是什么模式下,common module永远都是apply 'com.android.library',本身也不存在什么独立运行,直接贴伪代码
apply plugin: 'com.android.library'
apply plugin: 'com.jakewharton.butterknife'

android {
    compileSdkVersion rootProject.ext.android.compileSdkVersion

    defaultConfig {
        minSdkVersion rootProject.ext.android.minSdkVersion
        targetSdkVersion rootProject.ext.android.targetSdkVersion

        testInstrumentationRunner rootProject.ext.android.AndroidJUnitRunner
        
        ....
    }

    buildTypes {
        debug {
            ....
        }
        release {
            ....
        }
    }

    compileOptions {
        sourceCompatibility rootProject.ext.android.compileOptions.sourceCompatibility
        targetCompatibility rootProject.ext.android.compileOptions.targetCompatibility
    }

    resourcePrefix rootProject.ext.module_common.resourcePrefix_name
    sourceSets {
        main {
            ....
        }
    }

    publishNonDefault true
    
    ....
}

dependencies {
    api fileTree(include: ['*.jar'], dir: 'libs')
    api rootProject.ext.dependencies.appcompat_v7
    api rootProject.ext.dependencies.design
    api rootProject.ext.dependencies.butterknife
    annotationProcessor rootProject.ext.dependencies.butterknife_compiler
    
    ....
}

业务组件

业务组件需要根据不同情况切换模式,代码
if (isAModuleAppliction.toBoolean()) {
    apply plugin: 'com.android.application'
} else {
    apply plugin: 'com.android.library'
}
apply plugin: 'com.jakewharton.butterknife'

android {
    compileSdkVersion rootProject.ext.android.compileSdkVersion



    defaultConfig {

        if (isAModuleAppliction.toBoolean()) {
            applicationId rootProject.ext.android.AModuleapplicationId
            multiDexEnabled rootProject.ext.android.multiDexEnabled
        }

        minSdkVersion rootProject.ext.android.minSdkVersion
        targetSdkVersion rootProject.ext.android.targetSdkVersion

        testInstrumentationRunner rootProject.ext.android.AndroidJUnitRunner

    }
    
    resourcePrefix rootProject.ext.module_a.resourcePrefix_name
    sourceSets {
        main {
            if (isAModuleAppliction.toBoolean()) {
                manifest.srcFile 'src/main/runalong/AndroidManifest.xml'
            } else {
                manifest.srcFile 'src/main/AndroidManifest.xml'
                //集成模式下排除runalong文件夹中的所有Java文件
                java {
                    exclude 'runalong/**'
                }
            }
        }
    }

    compileOptions {
        sourceCompatibility rootProject.ext.android.compileOptions.sourceCompatibility
        targetCompatibility rootProject.ext.android.compileOptions.targetCompatibility
    }

    publishNonDefault true
    
    ....
}

dependencies {
    implementation fileTree(include: ['*.jar'], dir: 'libs')

    implementation project(':common')

    annotationProcessor rootProject.ext.dependencies.butterknife_compiler
    annotationProcessor rootProject.ext.dependencies.arouter_compiler
    ....
}

关于组件化混淆

一般关于组件化混淆有两种做法

  • 直接使用app空壳工程中的混淆规则,集成模式下一旦app空壳开始混淆,其他依赖的组件都会默认开启混淆。
  • 各自组件使用各自的混淆规则,需要有比较好的管理
选择第二种,需要在**build.gradle**中添加如下
release{
        consumerProguardFiles   'proguard-rules.pro'
}
业务组件中的混淆规则对app空壳工程是不构成影响的,所以就只存在该组件相关的混淆规则,共有的可以选择放在common组件或者app空壳中

总结

  • 组件化相较于单一工程,在组件模式下可以提高编译速度,方便单元测试,提高开发效率。
  • 开发人员分工更加明确,基本上做到互不干扰。
  • 业务组件的架构也可以自由选择,不影响同伴之间的协作。
  • 降低维护成本,代码结构更加清晰。

组件化其实并不复杂,复杂的是,我们开发者为了更加容易区分功能业务,把它解耦得更彻底,导致某些地方和以往的有所偏差,需要深入浅出的了解后才能处理,
这个个人认为跟mvc到mvp再到mvvm的发展历程道理是一样的,一样是为了解耦,写更多的东西,慢慢完善趋于稳定,所以离开舒适区,当然是要复出代价的。
组件化每个人的理解可能都会不同,我这边也需要慢慢完善,毕竟步子大了扯到蛋,当然这也不是组件化的最终形态,比如,你可以将组件上传私有maven,然后引用到项目上等等。。

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