Android组件化入门:一步步搭建组件化架构

一、前言

最近由于业务需求变动,有考虑采用组件化架构进行开发,这方面我以前没有接触过。关于组件化的文章不少,各方大神更是提出了各类的组件化方案,我也看了不少相关文章。可是学习新东西看的再多,不如动手作一次,先不考虑复杂的东西,先动手作个简单的Demo更有助于理解组件化的思想。组件化相关理论知识这里就很少讲了,想要了解的能够本身去搜或者去看Android组件化方案这篇文章。废话很少说,直接动手开码。java

二、搭建组件化Demo

先打开Android Studio新建一个项目。android

步骤一: 新建config.gradle,统一管理build.gradle中的相关内容

而后在项目目录下新建一个config.gradle文件。 git

接着在这个文件内添加以下代码:

ext {
    //表示是做为module仍是application
    isModule = false
    //applicationId版本好sdkVersion统一管理
    android = [
            compileSdkVersion        : 28,
            buildToolsVersion        : 28,
            applicationId            : "com.example.componenttestdemo",
            minSdkVersion            : 19,
            targetSdkVersion         : 28,
            versionCode              : 1,
            versionName              : "1.0",
            testInstrumentationRunner: "android.support.test.runner.AndroidJUnitRunner"
    ]

    //版本号
    def APPCOMPAT_V7_VERSION = "28.0.0"
    def CONSTRAINT_LAYOUT_VERSION = "1.1.3"

    //三方库统一管理
    dependencies = [
            appcompatV7     : 'com.android.support:appcompat-v7:' + APPCOMPAT_V7_VERSION,
            constraintLayout: 'com.android.support.constraint:constraint-layout:' + CONSTRAINT_LAYOUT_VERSION
    ]
}
复制代码

由于咱们知道项目使用组件化架构后,单一模块Module能够做为单个Application运行,同时也能够在整个主Application中做为一个Module运行。因此在config.gradle中先定义一个isModule来区别这两种状况,组件化以后能够经过修改这个值来切换这两种状况的使用。剩下就是对applicationId、版本号、sdkVersion和三方库等进行统一管理。github

接着修改app下的build.gradle里设置内容安全

将原来的compileSdkVersionapplicationIdminSdkVersionversionCode和三方库等替换成对应config.gradle中定义的值。bash

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.testInstrumentationRunner
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
}
dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation rootProject.ext.dependencies.appcompatV7
    implementation rootProject.ext.dependencies.constraintLayout

}
复制代码

最后还要在项目目录下的build.gradle中添加一行:服务器

apply from : "config.gradle"
复制代码

而后点击 Sync Now同步。最后在进行下一步前,新建一个MyApplication,在 AndroidManifest设置name属性。

步骤二:建立Main模块,搬空app壳工程

咱们知道组件化中须要一个app壳工程,这个壳工程中不处理任何业务,就只是一个空壳,由它将所须要的各个组件模块组合起来,构成一个完整的应用。而如今项目中的app仍是存在默认的入口Activity的,因此要新建一个ModuleMain将默认的MainActivity和其布局文件搬过去。markdown

接着进入app的AndroidManifest文件将注册Activity的相关代码也搬到ModuleMain模块的AndroidManifest中去,只留下application标签。架构

这里注意组件化项目中每一个Module都会有本身的AndroidManifest文件,最后打包时会将这些文件合并成一个文件,因此会出现application标签中的属性重复问题,要在app的AndroidManifest文件中添加以下两行代码:app

xmlns:tools="http://schemas.android.com/tools" 
复制代码
tools:replace="android:name,android:label,android:icon, android:theme,android:allowBackup"
复制代码

这里的namelabeliconthemeallowBackup均可能会有重复,因此所有写上之间用逗号隔开。完整AndroidManifest以下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.sy.modulesimpledemo"
    xmlns:tools="http://schemas.android.com/tools"    >

    <application
        android:name=".application.MyApplication"
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        tools:replace="android:name,android:label,android:icon, android:theme,android:allowBackup"
        android:theme="@style/AppTheme">
    </application>

</manifest>
复制代码

接着app壳工程中只剩刚修改的build.gradle还没删减,在删减前先将app中build.gradle的内容复制覆盖到Main模块的build.gradle中,而且还要作部分修改。由于单个组件能够做为一个组件模块被app壳工程组合使用,也能够单独做为一个application使用。因此要根据config.gradle中定义的isModule来判断是做为Module仍是Applicaition。一样还有做为Module是不须要applicationId的而做为应用则是须要的。

//经过isModule来判断是application仍是module
if (rootProject.ext.isModule) {
     apply plugin: 'com.android.library'
} else {
    apply plugin: 'com.android.application'
}
android {
    compileSdkVersion rootProject.ext.android.compileSdkVersion
    defaultConfig {
        //是application才须要applicationId
        if (!rootProject.ext.isModule) {
            applicationId "com.example.sy.moduledmain"
        }
        minSdkVersion rootProject.ext.android.minSdkVersion
        targetSdkVersion rootProject.ext.android.targetSdkVersion
        versionCode rootProject.ext.android.versionCode
        versionName rootProject.ext.android.versionName
        testInstrumentationRunner rootProject.ext.android.testInstrumentationRunner
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
    }
}

dependencies {
    implementation fileTree(include: ['*.jar'], dir: 'libs')
    implementation rootProject.ext.dependencies.appcompatV7
    implementation rootProject.ext.dependencies.constraintLayout
}
复制代码

这时modulemain中的AndroidManifest会提示资源文件的缺乏,这时先将app中的对应文件复制到modulemain里来。

步骤三 :新建ApplicationAndroidManifest文件

在app壳工程和ModuleMain中分别新建一个Application,由于Moudule也是须要能够单独运行的。

接着在ModuleMain里新建 AndroidManifest文件,由于做为 ModuleApplication是会有不同的因此要作区分,在main目录下新建module文件夹和application文件夹分别存放两个状况下的 AndroidManifest文件,将原来的 AndroidManifest文件拖到module下,再拷贝一份到application下。拷贝完了记得在两个 AndroidManifestapplication标签下设置name属性。

接着再build.gradle中添加以下代码,用来分别在两种状况下指定使用哪一个AndroidManifest

sourceSets {
        main {
            if (rootProject.ext.isModule) {
                manifest.srcFile 'src/main/module/AndroidManifest.xml'
            } else {
                manifest.srcFile 'src/main/application/AndroidManifest.xml'
                java {
                    //排除java/module文件夹下的全部文件
                    exclude '*module'
                }
            }
        }
    }
复制代码

而后再到app的build.gradle中在dependencies内添加如下代码,用来引入ModuleMain模块。

if (rootProject.ext.isModule) {
    implementation project(":modulemain")
}
复制代码

如今能够再次点击Sync Now等同步结束后,虽然项目中只有一个壳工程和一个主Module,可是已能够看到组件化的雏形。此时已经能够经过修改config.gradle里的isModule的值,进行ApplicationModule两种模式的切换,将ModuleMain做为app 的模块运行或者是单独做为一个应用运行了。

步骤四:新建其余组件Module和解决资源文件冲突

接着按照新建ModuleMain的步骤重复新建其余业务Module,这里我新建了3个Module,业务A:ModuleA与业务B:ModuleB和一个BaseModule。其中BaseModule主要存放一些基础类和工具类,只作为Module为上层业务模块提供服务。

接下来解决资源文件冲突的问题,进入ModuleMain的 build.gradle添加下面这行代码,为资源文件命名规范一个统一开头:

resourcePrefix "modulemain_"
复制代码

添加后起名是没按照规范Android Studio就会有一个提示:

按要求修改文件名后提示消失。

步骤五:使用ARouter进行组件间通讯

接下来就要处理组件间的通讯问题,采用阿里的ARouter。按照文档集成ARouter。 首先在defaultConfig下添加以下代码:

javaCompileOptions {
            annotationProcessorOptions {
                arguments = [AROUTER_MODULE_NAME: project.getName()]
            }
        }
    }
复制代码

再引入ARouter依赖:

implementation rootProject.ext.dependencies.arouter
implementation rootProject.ext.dependencies.arouterCompiler
复制代码

最后在Application中初始化ARouter

if (isDebug()) {           // 这两行必须写在init以前,不然这些配置在init过程当中将无效
    ARouter.openLog();     // 打印日志
    ARouter.openDebug();   // 开启调试模式(若是在InstantRun模式下运行,必须开启调试模式!线上版本须要关闭,不然有安全风险)
}
ARouter.init(mApplication); // 尽量早,推荐在Application中初始化
复制代码

这样ARouter就集成好了,接着在MoudleA和ModuleB中新建两个Activity,而后使用ARouter进行页面跳转。

ModuleMain中MainActivity.java:

public class MainActivity extends BaseActivity  {
    /**
     * toA
     */
    private Button mModulemainA;
    /**
     * toB
     */
    private Button mModulemainB;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.modulemain_activity_main);
        initView();
        initEvent();
    }

    private void initEvent() {
        mModulemainA.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                ARouter.getInstance().build(ARouterPath.PATH_MOUDULE_A).navigation();
            }
        });

        mModulemainB.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                ARouter.getInstance().build(ARouterPath.PATH_MOUDULE_B).withString("key","传递的数据").navigation();
            }
        });
    }

    private void initView() {
        mModulemainA = (Button) findViewById(R.id.modulemain_a);
        mModulemainB = (Button) findViewById(R.id.modulemain_b);
    }

}
复制代码

ModuleA中ModuleAActivity.java:

@Route(path = ARouterPath.PATH_MOUDULE_A)
public class ModuleAActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.modulea_activity_module_a);
    }
}
复制代码

ModuleB中ModuleBActivity.java:

@Route(path = ARouterPath.PATH_MOUDULE_B)
public class ModuleBActivity extends AppCompatActivity {
    @Autowired(name = "key")
    String data;
    /**
     * TextView
     */
    private TextView mTextViewB;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.moduleb_activity_module_b);
        ARouter.getInstance().inject(this);
        initView();
    }
    private void initView() {
        mTextViewB = (TextView) findViewById(R.id.textViewB);
        mTextViewB.setText(data);
    }
}
复制代码

运行效果:

这里看到模块之间界面通讯正常,而且单个业务模块也能够单独运行,这样一个最基本的组件化架构Demo差很少就完成了。

三、将Module做为远程maven仓库

在开发中,可能会把一些公用Module传到私有服务器上,而后在项目中直接依赖使用。下面就将Module上传到Github做为远程maven仓库,在项目直接引用。首先新建一个项目,建立一个UtilModule。

将原来项目中的工具类移到UtilModule中,接着在UtilModule的build.gradle中添加如下代码:

apply plugin: 'maven'
uploadArchives {
    repositories.mavenDeployer {
        def mavenDirPath = file('\\Users\\sy\\AndroidProjects\\UtilModule') // 本地存放地址
        repository(url:"file://${mavenDirPath.absolutePath}")
        pom.project {
            groupId "com.example.utilmodule" // 包名
            artifactId "utilmodule" // module的名字
            version "1.0.0" // 版本号
        }
    }
}
复制代码

而后点击gradle中的uploadArchives

进入设置的目录查看,aar已经打包好了。

接着打开Github建立一个新仓库:

按照Github上的命令,将本地打包好的UtilModule上传到Github上:

上传完成后将仓库地址复制下来,将其中的 github.com部分修改成 raw.githubusercontent.com再在结尾加上 /master表示是主分支,添加到项目中的 build.gradle中。

接着在到Module的 build.gradle中添加依赖:

utilmodule      : 'com.example.utilmodule:utilmodule:' + UTIL_MODULE_VERSION
复制代码
implementation rootProject.ext.dependencies.utilmodule
复制代码

这里就是以前设置的包名:Module名:版本号。SyncNow以后删除原来项目中的工具类,而后在代码里使用远程仓库的工具类测试:

运行打印日志:

D/com.example.modulemain.MainActivity: onCreate:false
复制代码

这说明远程仓库依赖成功已经能正常使用其中的类和方法。

四、总结

这篇文章主要是记录下我初识组件化,搭建组件化Demo的过程,Demo主要对于我对组件化思想的理解和体验仍是颇有帮助的,Demo中还有不少没考虑到的地方,好比Application的动态配置合并、Fragment、组件化的混淆等等,也是我正在学习的问题。这篇文章主要供和我同样对组件化这块不太了解的新手作参考,但愿能对新手有所帮助。

最后再贴上几个总结得比较好的开源组件化方案的连接,也是我后面准备研究学习的方案:
多个维度对比一些有表明性的开源android组件化开发方案
AppJoint方案
ArmsComponent方案

补一下文中Demo源码地址:github.com/syDeveloper…

相关文章
相关标签/搜索