0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

剖析Android项目组件化重构架构

jf_78858299 来源:小余的自习室 作者:小余的自习室 2023-03-30 10:41 次阅读

1.组件化重构效果

这里先看下我们重构前后的框架图比较:

重构前:

图片

传统代码架构.png

重构后

图片

组件化代码架构.png

  • ft_xxx表示业务层模块 lib_xxx表示基础库模块

重构后的架构图如下

图片

服务接口调用.png

重构前的代码业务封装在宿主app中,业务耦合严重,如果修改一个业务模块,需要对整个app进行完整测试,测试工作量巨大

重构后,我们只需要对单一app进行独立调试即可。

重构后的框架结构:所有的业务组件之间通讯都通过ft_base_service进行通讯

2.组件化重构准则

  • 1.单一业务可以单独调试,也可以作为lib提供给宿主app使用
  • 2.同一级别的模块不允许直接调用,比如我们的ft_home组件不允许直接调用ft_login组件,不然组件化的意义就不存在了
  • 3.组件间通讯不能直接使用显示的class文件跳转,可以考虑很用ARouter框架进行解耦
  • 4.每个组件可打包为aar或者jar上传到maven私服,宿主使用的时候,直接引用私服中aar包即可

能做到以上几点,你的app就可以称为一个组件化框架的app了。

3.组件化重构思路

图片

重构思路.png

    1. :拆代码,拆资源,拆构建

      由于所有业务和资源都耦合在宿主app中,所以需要将代码和资源拆开到对应模块中

      当然我们的构建build.gradle也需要拆分到不同模块中

    1. :对外提供接口

      组件化之间不能直接通讯,需要使用暴露接口的方式对外通讯

    1. :反复测试

      重构后代码,需要反复测试,防止出现意想不到的bug

4.组件化重构过程

这里我以登录业务ft_login为例子:

1. 步骤1 :首先新建一个业务模块ft_login,然后在宿主app中将登录功能相关联的代码和资源抽离到ft_login

2. 步骤2 :将和登录构建相关的依赖分配到ft_login构建中。

3. 步骤3 :单独调试功能实现

  • 3.1:在gradle.properties中创建一个全局变量:isRunAlone=true
  • 3.2:在build.gradle中:
if(isRunAlone.toBoolean()){
    apply plugin:'com.android.application'
}else{
    apply plugin:'com.android.library'
}

android {
    compileSdkVersion 33
    buildToolsVersion "33.0.0"

    defaultConfig {
        if(isRunAlone.toBoolean()){
            applicationId 'com.anna.ft_login'
        }
        ...
    }
    sourceSets {
        main {
            java {
                srcDirs = ['src/main/java']
            }
            resources {
                srcDirs = ['src/main/res']
            }
            aidl {
                srcDirs = ['src/main/aidl']
            }
            manifest {
                if(isRunAlone.toBoolean()){
                    srcFile 'src/main/manifest/AndroidManifest.xml'
                }else {
                    srcFile 'src/main/AndroidManifest.xml'
                }
            }
        }
    }
}
def dependList = [rootProject.depsLibs.okhttp,
                  rootProject.depsLibs.gson,
                  rootProject.depsLibs.appcompact,
                  rootProject.depsLibs.design,
                  rootProject.depsLibs.eventbus,
                  rootProject.depsLibs.arouterapi,
                  ':lib_network',':lib_common_ui',':ft_base_service']


dependencies {
    if(!isRunAlone.toBoolean()){
        dependList.each { String depend ->
            depend.startsWithAny(':lib',':ft')? compileOnly(project(depend)):compileOnly(depend){
                switch (depend){
                    case rootProject.depsLibs.arouterapi:
                        exclude group: 'com.android.support'
                        break;
                }
            }
        }
    }else {
        dependList.each { String depend ->
            depend.startsWithAny(':lib',':ft')? implementation(project(depend)):implementation(depend) {
                switch (depend) {
                    case rootProject.depsLibs.arouterapi:
                        exclude group: 'com.android.support'
                        break;
                }
            }
        }
    }
    //arouter注解处理器
    annotationProcessor rootProject.depsLibs.aroutercompiler

    testImplementation 'junit:junit:4.+'
    androidTestImplementation 'androidx.test.ext:junit:1.1.3'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0'

}

单独调试状态下注意四点

  • 1.引用application插件
  • 2.引入applicationId
  • 3.引入不同给的sourceSets构建路径
  • 4.引入的库单独调试状态下需要使用implementation导入,不能使用compileOnly

实现上面四点, 只要打开isRunAlone就可作为一个单独app运行了

4.步骤4:组件间通讯

这里,我们引入一个ft_base_service模块,这个模块用来实现组件间通讯用,需要调用别的业务模块都需要使用这个模块才能通讯、

业务模块与ft_base_service之间通讯使用的是路由ARouter

关于ARouter的使用可以参考这篇文章:

Android开源系列-组件化框架Arouter-(一)使用方式详解

  • 1.创建ft_base_service,在这个模块中:创建一个LoginService接口继承IProvider

引入ARouter依赖

android {
    javaCompileOptions {
        annotationProcessorOptions {
            arguments = [AROUTER_MODULE_NAME: project.getName(), AROUTER_GENERATE_DOC: "enable"]
        }
    }
}
//arouter核心api
implementation rootProject.depsLibs.arouterapi
//arouter注解处理器
annotationProcessor rootProject.depsLibs.aroutercompiler

创建LoginService:

public interface LoginService extends IProvider {
    boolean hasLogin();
    void login(Context context);
}
  • 2.在ft_login业务模块中实现LoginService接口

注意这里因为使用了ARouter注解,所以也需要引入ARouter依赖

@Route(path = "/login/login_service")
public class LoginServiceImpl implements LoginService {
    Context context;
    @Override
    public boolean hasLogin() {
        return UserManager.getInstance().hasLogined();
    }

    @Override
    public void login(Context context) {
        LoginActivity.start(context);
    }

    @Override
    public void init(Context context) {
        Log.d("TAG","LoginServiceImpl is init");
    }
}
  • 3.在ft_base_service模块中对LoginService接口进行依赖注入
public class LoginImpl {

    @Autowired(name = "/login/login_service")
    public LoginService mLoginService;
    private static LoginImpl mLoginImpl = null;
    public static LoginImpl getInstance() {
        if (mLoginImpl == null) {
            synchronized (LoginImpl.class) {
                if (mLoginImpl == null) {
                    mLoginImpl = new LoginImpl();
                }
                return mLoginImpl;
            }
        }
        return mLoginImpl;
    }

    private LoginImpl(){
        ARouter.getInstance().inject(this);
    }
    public boolean hasLogin(){
        return mLoginService.hasLogin();
    }
    public void login(Context context){
        mLoginService.login(context);
    }

}

笔者使用了一个 单例类LoginImpl ,在构造器中对LoginService依赖注入

ARouter.getInstance().inject(this);

然后宿主app或者其他模块引用登录业务功能时,需要依赖ft_base_service模块,并使用LoginImpl的接口即可。

这里要说明下,平时我们使用的四大组件跳转也可以使用这个方式来处理,在服务接口中定义跳转接口即可。当然也可以使用Arouter的Activity跳转方式或者Fragment实例获取方式

  • 5.代码打包aar上传到maven私服

关于这块maven私服更多内容可以参考这篇文章:

Gradle筑基篇(六)-使用Maven实现组件化类库发布

这里我们封装了一个通用组件发布库:

apply plugin: 'maven'


uploadArchives {
    repositories {
        mavenDeployer {
            // 是否快照版本
            def isSnapShot = Boolean.valueOf(MAVEN_IS_SNAPSHOT)
            def versionName = MAVEN_VERSION
            if (isSnapShot) {
                versionName += "-SNAPSHOT"
            }
            // 组件信息
            pom.groupId = MAVEN_GROUP_ID
            pom.artifactId = MAVEN_ARTIFACTID
            pom.version = versionName

            // 快照仓库路径
            snapshotRepository(url: uri(MAVEN_SNAPSHOT_URL)) {
                authentication(userName: MAVEN_USERNAME, password: MAVEN_USERNAME)
            }
            // 发布仓库路径
            repository(url: uri(MAVEN_RELEASE_URL)) {
                authentication(userName: MAVEN_USERNAME, password: MAVEN_USERNAME)
            }

            println("###################################"
                    + "\\nuploadArchives = " + pom.groupId + ":" + pom.artifactId + ":" + pom.version + "." + pom.packaging
                    + "\\nrepository =" + (isSnapShot ? MAVEN_SNAPSHOT_URL : MAVEN_RELEASE_URL)
                    + "\\n###################################"
            )
        }
    }
}

然后在对应的组件下面引用:

apply from:file('../maven.gradle')

发布的时候直接在Gradle面板中点击uploadArchives任务即可

图片

task面板.png

经过上面几个步骤就基本完成了login组件的封装并发布,且对外提供了login组件接口 其他组件也是按照上面的逻辑进行重构

更多详细信息可以自己拿到项目源代码查看。

5.组件化重构总结

组件化不仅是一种架构,更是一种思想,架构是可以变得,但是核心思想却是统一的,在拆分代码的时候,要注意模块的颗粒度,不是颗粒度越小就越好,模块分离的好,后期对组件改造会有很大帮助, 关于组件化的文章就讲到这里,组件化重构的项目已经上传到Github。后面会出一期插件化的项目改造。敬请期待。

** demo地址: https://github.com/ByteYuhb/anna_music_app **

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Android
    +关注

    关注

    12

    文章

    3717

    浏览量

    124839
  • APP
    APP
    +关注

    关注

    32

    文章

    1465

    浏览量

    71206
  • 代码
    +关注

    关注

    28

    文章

    3933

    浏览量

    65598
收藏 人收藏

    评论

    相关推荐

    Android的系统架构包括哪些部分?

    Android以后,再来说说Android的系统架构包括哪些部分。Android的系统架构和其操作系统一样,采用了分层的架构Android分为四个层,从高层到低层分别是应用程序层、应用程序框架层、系统运行库层
    发表于 12-04 14:11

    【悬赏100块】如何实现FPGA可重构计算(Android平台)

    Android平台上实现可重构计算:简单说,就是实现应用程序把一部分计算密集型的任务交给FPGA来计算,把FPGA作为CPU的一个
    发表于 05-20 20:03

    STM32库函数架构剖析

    架构剖析
    发表于 06-20 17:08

    [资料分享]+Android内核剖析

    Android内核剖析一、看威武霸气的封面作者:柯元旦 二、读读简介,看看适合你吗? 本书详细分析了Android内核的内部机制,包括窗口管理系统、Activity管理系统
    发表于 09-26 10:46

    Android手机操控ARM开发板外围硬件设备【创科之龙】原创

    架构分布剖析零死角玩转Android下_如何进行Android源码编译Makefile剖析 零死角玩转Android启动_开机启动流程_总体架构剖析零死角玩转Android启动
    发表于 01-12 22:31

    Android RIL架构简介

    架构说起—Android架构从手机的架构说起—Android给客户的价值从RIL的架构说起—RIL的位置从RIL的架构说起—RIL架构从RIL的架构说起—RIL的作用1、承上启下2、封装AP3、无缝
    发表于 09-25 15:46

    初探Android系统整体架构

    Android系统庞大且错综复杂,今天小编将带领大家初探Android系统整体架构,一窥其全貌。引言本文作为Android系统架构的开篇,起到提纲挈领的作用,从系统整体架构角度概要讲解
    发表于 08-20 06:32

    XML在可重构制造执行系统组件管理中的应用

    组件技术研究和开发可重构制造执行系统(RMES),需要解决大量不同功能业务组件的维护问题。本文基于XML-Schema 机制设计了一套标准描述方法,来定义RMES 组件的自然属性、继
    发表于 08-24 11:31 7次下载

    深入剖析Android消息机制

    剖析Android消息机制
    发表于 01-22 21:11 11次下载

    关于iOS组件的分析

    组件的目的是什么? 最近一两年很多人都想在项目里面搞组件,觉得搞组件好,却鲜有人知道组件好在哪里?组件的目的是什么?个人觉得组件主要有两个目的: 1.实现
    发表于 09-25 10:28 0次下载
    关于iOS<b>组件</b><b>化</b>的分析

    基于AndroidCrop进行鸿蒙的开源图片裁剪组件

    项目是基于开源项目 AndroidCrop 进行鸿蒙的移植和开发的,可以通过项目标签以及github地址( https://github.com/jdamcd/android-crop )追踪到
    发表于 03-23 09:53 2次下载

    华为BBF CloudCO架构标准为运营商重构宽带网络架构提供了关键的参考

    架构标准为运营商重构宽带网络架构提供了关键的标准参考,是迈向云网络的关键一步。
    发表于 10-25 09:45 2469次阅读

    重构计算架构将引领未来芯片的市场发展

    重构计算架构AI芯片实现量产的消息在业内迅速传开,可重构计算架构芯片再次引发一波讨论的热潮。
    的头像 发表于 12-16 15:55 2839次阅读

    重构架构技术的快速发展,未来将应用于太空

    。赛灵思太空产品系统架构师Minal Sawant表示,航天行业的主要挑战之一是用于各种项目的体系结构,多任务的灵活架构平台是如今的一个挑战。
    发表于 05-22 10:35 437次阅读

    基于安卓平台的图片裁切组件已实现鸿蒙迁移和重构

    组件 crop_image_layout,实现了鸿蒙迁移和重构,代码已经开源,目前已经获得了很多人的 Star 和 Fork ,欢迎各位下载使用并提出宝贵意见
    的头像 发表于 11-10 09:22 1140次阅读
    基于安卓平台的图片裁切<b>组件</b>已实现鸿蒙<b>化</b>迁移和<b>重构</b>