青团社拆包依赖-QDepend

青团社业务发展愈发多样,Android使用模块化设计进行业务的解耦,在代码都解耦上我们已经有一套方案,但在工程的模块化设计上,还缺少一个成熟的方案。

在前期模式过程中,我们在component_task模块上率先进行了尝试,结合当时的需求和场景,我们使用率git submodule和pin工程来实现task工程的管理。

  • git submodule 将代码都版本管理从主工程中抽离,实现在多个主工程中代码的同步
  • pin工程 细化业务粒度,实现不同flavor下的特定代码实现

改设计上开发一段时间后,git submodule暴露了一些问题:

  • 学习成本,需要学习git submodule的一些命令和思想,尤其是commit id的概念
  • 代码稳定性不可靠,主工程对task的依赖实际是通过commit id进行依赖,多人开发下会出现下拉代码,commit id并不会自动去更新本地的task工程,因此可能会导致未处理的情况下提交代码覆盖了线上的最新commit id

虽然在生产环境下会对commit id进行校验,但独立主工程的版本管理,版本号的缺失,以及维护成本确实对开发不够友好。

为了提高代码稳定性,设计了一套新的依赖管理,在效率上较submodule低一些,但代码稳定性有很大的保障,责任划分也更加明确。

gradle中的依赖

gradle中主要有三种依赖类型

  • module
  • project
  • file

module依赖最简洁,只需要将对应的仓库路径添加进去即可:

    implementation 'com.google.code.gson:gson:2.8.5'

project需要配置setting和build文件

depend1.png

在开发期会倾向于使用project对业务模块进行开发,在打包或者不需要开发的模块使用module依赖方式,手动配置需要改动多个文件,效率低。

QDepend

一键切换依赖模式,确保生产环境aar版本依赖,开发环境动态依赖

对整个切换逻辑进行封装,作为一个plugin添加到maven中,方便实用。

depend2.png

Android Stuido中传统的多module工程结构如下:

├── app
├── component_jobs
├── component_login
├── component_me

QDepend改造后的工程,将原有的module抽离,工程中只含有一个app主module,内部提供maven依赖各个模块

├── app

module作为单独的component库,有独立的git管理,默认的目录结构是与工程平级

├── DrinkWater
│   └──  app
├── component_jobs
├── component_login
├── component_me

在切换maven依赖和本地project依赖时,只需配置module_map.properties中的开关位即可

使用

在一个新的工程中,我们需要手动适配QDepend插件,进行如下操作:

  1. 在gradle/depend目录下添加module_map.json
  2. 工程目录下添加module_map.properties
  3. setting.gradle添加com.qtshe.setting插件
  4. app/build.gradle添加com.qtshe.depend插件

改造流程

1.添加module_map.json

module_map.json按照特定的数据格式配置,添加需要动态控制的module

{
  "libs": [
    {
      "name": "homepage",
      "group": "com.qtshe.mobile.component",
      "path": "../component_homepage"
    },
    {
      "name": "me",
      "group": "com.qtshe.mobile.component",
      "path": "../component_me"
    }
  ]
}

其中:

  • name: 对应的module project名
  • group: 在maven中group字段
  • path: 本地的module路径
qdepend1.png

比如gson的依赖:

 implementation 'com.google.code.gson:gson:2.8.5'

name:com.google.code.gson

group:gson

2.添加module_map.properties

module_map.properties作为动态依赖的开关,把需要的module name设置为true即可,注意必须在json中存在:

homepage = false
me = true

以上会对me模块替换成本地依赖

qdepend2.png

3.setting.gradle修改

需要引入plugin,注意setting不属于build script,所以要单独添加classpath

include ':QtsCustomer'

buildscript {
    repositories {
        maven { url "http://xxxx/nexus/content/repositories/releases/" }
        maven { url "http://xxxx/nexus/content/repositories/snapshots/" }
        google()
        jcenter()
    }

    dependencies {
        classpath 'com.qtshe.mobile:dependencies:1.1.1-SNAPSHOT'
        classpath 'com.google.code.gson:gson:2.8.5'

    }
}
apply plugin:"com.qtshe.setting"
qdepend3.png

4.build.gradle

因为要使用插件,所以需要在根目录的build.gradle中添加classpath

buildscript {
    repositories {
        maven { url "http://xxxx/nexus/content/repositories/releases/" }
        maven { url "http://xxxx/nexus/content/repositories/snapshots/" }
        google()
        jcenter()
        ……
    }
    dependencies {
        ……
        classpath 'com.qtshe.mobile:dependencies:1.1.1-SNAPSHOT'
        classpath 'com.google.code.gson:gson:2.8.5'

    }
}

同时在build.gradle中应用插件

apply plugin: "com.qtshe.depend"
qdepend4.png

适配好后只需要修改module_map.properties就可以实现切换

依赖库的管理

一个完整的maven依赖包含group,module,version三部分。

模块化开发中规范如下:

  1. 所有都component都以com.qtshe.mobile.component作为group。
  2. module命名以flavor-module格式,如趣喝水的login模块:drinkwater-login
  3. 开发版:DEV-SNAPSHOT;正式版:verison_name.xx 如趣喝水login:1.0.0.1

开发模式

开发阶段会涉及到频繁的修改代码,本地开发时可以通过QDepend切换本地模式即可,但实际情况往往是多人协助开发,以及提测时jenkins上不断的修复bug更新代码,因此开发模式可以将依赖改为snapshot模式,以下是趣喝水开发阶段的component依赖:

    implementation 'com.qtshe.mobile.component:drinkwater-common:1.0.6'
    implementation ('com.qtshe.mobile.component:drinkwater-login:DEV-SNAPSHOT'){
        exclude group:'com.qtshe.mobile.component', module:'drinkwater-common'
    }
    implementation ('com.qtshe.mobile.component:drinkwater-me:DEV-SNAPSHOT'){
        exclude group:'com.qtshe.mobile.component', module:'drinkwater-common'
    }
    implementation ('com.qtshe.mobile.component:drinkwater-jobs:DEV-SNAPSHOT'){
        exclude group:'com.qtshe.mobile.component', module:'drinkwater-common'
    }

SNAPSHOT会在每次下拉代码时拉取最新的依赖,保证协助开发的同步和打包机出包代码的时效性

如何生成SNAPSHOT?

工程中已经集成了nnaven_publish.gradle,在rootProject下执行如下命令即可更新本地依赖的module到仓库中:

./gradlew qpublish

该脚本需要结合component中的maven.properties文件:

# module的版本
MODULE_VERSION=1.0.6
# module的maven名
MODULE_ID=common
# 标记不生成SNAPSHOT
IS_LIB=true

其中IS_LIB是为了剔除common等lib,IS_LIB标记后不会生成SNAPSHOT,标准的component工程不需要添加。

为了保证snapshot刷新的时效性,工程配置了cacheChangingModulesFor刷新时间:

    configurations.all{
        resolutionStrategy.cacheChangingModulesFor 0, 'seconds'
    }

实时刷新可能会影响编译速度,不需要的话可以在rootProject的properties中修改标志位:

# is refrsh snapshot
REFRESH_SNAPSHOT = false

正式环境

在提测接近结束时,需要将更新过的component进行发版。具体步骤如下:

1.版本需要指定版本号,更新component下的maven.gradle文件:

# module的版本
MODULE_VERSION=1.0.0.1
# module的maven名
MODULE_ID=common

将MODULE_VERSION指定为最新版本,版本号的规则约定如下:

"${version_name}.x"

比如趣喝水初版版本为1.0.0,对应的component版本号为:1.0.0.1到1.0.0.99,期间可能还存在代码修改的情况,因此预留两位用于更新

2.提交代码到远程仓库

将component的代码更新到release/xxx上,xxx为指定的flavor,如release/drinkwater。

在QDepend中 component的release/xxx类似git flow中的master,可以添加权限,以及用于code review

3.ModuleToMaven上执行打包命令

通过jenkins,选择需要更新的component和flavor,然后build

jenkins_module_to_maven.png

4.更新工程的依赖

将项目中对component的依赖更新到固定版本:

implementation 'com.qtshe.mobile.component:drinkwater-login:1.0.0.1'

5.发版文档更新

每个项目都有有自己的wiki文档,开发负责人需要发版前进行更新

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