android 应用的组件化

来源:互联网 发布:项目经理薪水知多少 编辑:程序博客网 时间:2024/06/06 11:04

android 应用的组件化

一、组件化的意义

代码级解耦我们可以通过MVP、MVVM等去实现,但是随着公司业务的逐步增多,项目体积也在不断增长,这时候就需要进行一种项目级的解耦了,把各个业务单独出一个组件,既方便调试又可以做到分工合作,否则一个项目无论前期设计多合理多优雅也会慢慢写成一坨屎的。。。

二、组件化的实现

组件化基本实现原理就是编译期间各个组件的依赖整合,apk还是一个apk。具体的实现思路就是将一个项目分离成一个容器APP(用来加载各个组件),一个基础Library(组件所需要的一些基类、公用第三方库和资源文件等),和多个组件Module(具体的业务实现)的形式。
好,下面开始简单的模拟下如何一步一步实现组件化。

三、Music!Action!

假设我们要有一个叫做Modules的项目,Modules有两个主要功能:音乐和读书,我们就先按这个去创建项目。

1、新建项目Modules

2、新建Library

右键工程,new->module->Android Library  名称就叫library好了在library里我们可以引入一些第三方库,比如Okhttp,Rxjava,Logger.Retrofit等等,也可以引入一些Utils工具类等等。library里的资源可以被依赖它的Moudle所引用的,包括string、color、drawable等等。

3、新建音乐Module和读书Module

右键工程,new->module->Phone & Tablet Module,module就叫music吧同理,新建一个叫book的module

4、更改gradle配置

每一个Module都需要两种形态,分别是:测试阶段可以单独运行的Application形态,以及发布版作为library融入app的library形态。为了区分这两种形态,按理说我们应该需要两套配置文件,为了方便起见,我们通过动态修改build配置文件来达到此目的。1)、在gradle.properties(Project Properties)里新建一个boolean值的变量isDebug用以标识组件的形态。代码如下:org.gradle.jvmargs=-Xmx1536m#将Module作为Application还是library, false->library  true->ApplicationisDebug=false2)修改music的build.gradle文件,内容如下:   if (isDebug.toBoolean()){    // 为true时表示此Module是一个独立的应用        apply plugin: 'com.android.application'    }else{    // 为false时表示此Module是一个library        apply plugin: 'com.android.library'    }android {    compileSdkVersion 25    buildToolsVersion "25.0.2"    defaultConfig {        // 作为library时不能有applicationId,只有作为一个独立应用时才能够如下设置        if (isDebug.toBoolean()){            applicationId "com.liwy.music"        }        minSdkVersion 14        targetSdkVersion 25        versionCode 1        versionName "1.0"        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"    }    buildTypes {        release {            minifyEnabled false            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'        }    }    sourceSets {        main {            if (isDebug.toBoolean()) {                manifest.srcFile 'src/main/debug/AndroidManifest.xml'                java.srcDirs += "src/main/debug/java/";            } else {                manifest.srcFile 'src/main/release/AndroidManifest.xml'                java.srcDirs += "src/main/release/java/";            }        }    }    // 为防止资源冲突,设置资源别名,music下所有的资源名称都要以“music_”开头,否则会报错的哦!    resourcePrefix "music_"}dependencies {    // 加载公用library    compile project(':library')}同理,按照此方式设置Book的build.gradle。3) 配置AndroidManifest.xml文件    由于不同形态加载不同的AndroidManifest.xml文件,切换至Project目录状态下,在src/main/下新建两个目录:debug和release,并分别拷入AndroidManifest.xml,两种形态下的AndroidManifest.xml的配置是不一样的哦,至少library形态下的.xml没有启动activity没有theme没有icon等等, 内容如下:    library状态下的AndroidManifest.xml文件内容:        <?xml version="1.0" encoding="utf-8"?>        <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.liwy.music">            <application                 android:allowBackup="true"                android:supportsRtl="true">                <activity android:name=".MusicActivity"></activity>            </application>        </manifest>  4)根据build.gradle里配置的资源别名,修改module下的资源名称,包括layout,drawable,string,color等属性,反正res文件夹下面的资源名称都改掉好了,反正不改会报错。。。  5)按2)-> 4)的步骤设置book。

5、在app里引入这两个Module

1)修改app的build.gradle文件dependencies {    if (!isDebug.toBoolean()) {        compile project(':music')        compile project(':book')    } else {        compile project(':library')    }}2)在App的MainActivity里增加两个按钮,并设置其点击事件,使其一个跳转至music下的MusicActivity,一个跳转至book下的BookAcitivity。package com.liwy.modules;import android.content.Intent;import android.support.v7.app.AppCompatActivity;import android.os.Bundle;import android.view.View;import com.liwy.book.BookActivity;import com.liwy.music.MusicActivity;public class MainActivity extends AppCompatActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        findViewById(R.id.btn_music).setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                Intent intent  = new Intent(MainActivity.this, MusicActivity.class);                startActivity(intent);            }        });        findViewById(R.id.btn_book).setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                Intent intent  = new Intent(MainActivity.this, BookActivity.class);                startActivity(intent);            }        });    }}

6、运行

如果没有搞错的话,是可以正常跳转至MusicActivity和BookActivity页面。如果搞错了的话,那你还犹豫什么...查去吧...肯定上面步骤有遗漏的地方...

7、引入路由框架

 细心的同学可能会发现一个问题,那就是在app的MainActivity为了跳转引用了book和music两个组件的页面类BookActivity和MusicActivity。如果此时我将gradle.properties(Project Properties)里的isDebug设置为true,表示两个组件都是独立的application了,而app也不会再去依赖这两个Module,那么问题就来了,在NainActivity里引入的类将无法找到路径,从而导致报错。 为了解决这个问题,达到组件间解耦的目的,我们引入一个开源的框架ActivityRouter。 1)在library的build.gradle里引入activityrouter: compile 'com.github.mzule.activityrouter:activityrouter:1.2.2' 2)在app,book和music的build.gradle里分别引入activityrouter的编译组件  annotationProcessor 'com.github.mzule.activityrouter:compiler:1.1.7' 3)在app的AndroidManifest.xml里加入如下代码:  <activity             android:name="com.github.mzule.activityrouter.router.RouterActivity"             android:theme="@android:style/Theme.NoDisplay">             <intent-filter>                 <action android:name="android.intent.action.VIEW" />                 <category android:name="android.intent.category.DEFAULT" />                 <category android:name="android.intent.category.BROWSABLE" />                   <!-- jump_scheme = web -->                 <data android:scheme="@string/jump_scheme" />             </intent-filter>         </activity> 4)在app,book和music各自的包下分别创建一个空类,在类名上方加入注解@Module("app"),如下: // 表示此类所在的Module的组件名称为app @Module("app") public class AppModule { } 5)在app所在的Module的Application类上增加注解:@Modules({"app","music","book"}) 如果你没有重写Application类,那就新建一个吧...别忘了在app的AndroidManifest.xml里引入哦! 6)配置页面类的跳转路径 此次先拿BookActivity试水吧,在类名上方增加注解:@Router("book") @Router("book") public class BookActivity extends AppCompatActivity {     @Override     protected void onCreate(Bundle savedInstanceState) {         super.onCreate(savedInstanceState);         setContentView(R.layout.book_activity_book);     } } 7)修改app的MainActivity下前往BookActivity页面的跳转方式,修改为:  findViewById(R.id.btn_book).setOnClickListener(new View.OnClickListener() {             @Override             public void onClick(View v) {                 Routers.open(MainActivity.this, Uri.parse("web://book"));             }         });   跳转路径的组成方式"web://book" = (AndroidManifest.xml里的jump_scheme值web) + :// + (页面Activity上增加的Router注解值book)

8、最终幻想

ok!反正我是完美运行啊,这样app里页面访问组件类的页面就无需引入其类路径了,传值也可通过注解配置进行传递,activityrouter的传值使用详情大家还是百度一下吧,网上资料还是很多的。好了,整个组件化的简单使用就介绍到这里,具体的运用还是要仁者见仁智者见智,尽情的翱翔吧!该下班吃饭了,走起!

参考文章如下:
http://blog.csdn.net/wenyiqingnianiii/article/details/52891761
http://mp.weixin.qq.com/s/NwbnjStNO_M4VWT6nHQcAg

需要源码的同学请使劲戳这里

0 0
原创粉丝点击