[关闭]
@asce1885 2015-06-20T03:43:45.000000Z 字数 5027 阅读 516

Android Studio分模块自动化构建实战

Android


@author ASCE1885的 Github 简书 微博 CSDN

最近在使用Android Studio+Gradle做一个基础框架SDK项目,该框架主要实现每个app都需要的基础能力,例如网络请求,图片缓存,json解析,日志记录等等。

众所周知,AndroidStudio中应该尽量使用Module来进行模块的划分,既能达到模块解耦的目的,也能在必要的时候轻松实现分模块打包,特别是在SDK项目中。那么什么是分模块打包呢?就是我们可以根据第三方使用者的需求,自动化的提供SDK的全量版本,部分功能版本以及最小功能版本等等。

我们的项目结构如下所示,每个功能独立成一个Module:

由于我们的模块都是纯代码的,没有包含资源文件,因此不是以aar包的形式而是使用jar包形式对外提供。顺便提一句,生成的aar包默认路径是:

  1. build/output/aar/

而jar包可以到如下路径寻找:

  1. build/intermediates/bundles/debug/classes.jar
  2. build/intermediates/bundles/release/classes.jar

Jar包的合并

从项目工程截图中可以看到,我们的project包含多个module,每个基础功能的module最终编译生成的都是一个classes.jar。因此project最终会生成一堆的jar包,而到了对外发布时,我们要提供一个单独的jar包出去,因此,就需要对jar包进行合并。很不幸,Android Studio没有提供这样的功能,因此只能靠自己写脚本调用jar命令来实现了,打开命令行terminal,输入jar,就可以打印出jar的用法,如下所示:

  1. guhaoxindeMacBook-Pro:~ guhaoxin$ jar
  2. 用法: jar {ctxui}[vfm0Me] [jar-file] [manifest-file] [entry-point] [-C dir] files ...
  3. 选项包括:
  4. -c 创建新的归档文件
  5. -t 列出归档目录
  6. -x 从档案中提取指定的 (或所有) 文件
  7. -u 更新现有的归档文件
  8. -v 在标准输出中生成详细输出
  9. -f 指定归档文件名
  10. -m 包含指定清单文件中的清单信息
  11. -e 为捆绑到可执行 jar 文件的独立应用程序
  12. 指定应用程序入口点
  13. -0 仅存储; 不使用情况任何 ZIP 压缩
  14. -M 不创建条目的清单文件
  15. -i 为指定的 jar 文件生成索引信息
  16. -C 更改为指定的目录并包含其中的文件
  17. 如果有任何目录文件, 则对其进行递归处理。
  18. 清单文件名, 归档文件名和入口点名称的指定顺序
  19. 'm', 'f' 'e' 标记的指定顺序相同。
  20. 示例 1: 将两个类文件归档到一个名为 classes.jar 的归档文件中:
  21. jar cvf classes.jar Foo.class Bar.class
  22. 示例 2: 使用现有的清单文件 'mymanifest'
  23. foo/ 目录中的所有文件归档到 'classes.jar' 中:
  24. jar cvfm classes.jar mymanifest -C foo/ .

使用jar命令,主要实现两个功能:

由于jar命令不能指定最终输出的目录,因此我们需要首先cd到用于存放解压后class文件的一个临时目录,然后依次对所有jar包进行解压操作,解压命令如下所示:

  1. jar -xvf ../../hfasynchttp/build/intermediates/bundles/debug/classes.jar

当所有的jar包都解压完毕后,接着执行压缩命令,这样就得到一个单独的jar包了:

  1. jar -cvfM AndroidHyperion_${version}_debug.jar .

分模块自动化构建

自动化构建包括本地构建和Jenkins构建两部分,本地构建主要用于开发自己调试使用,Jenkins构建主要用于测试,产品等取包以及跑Monkey使用。

本地构建

本地构建脚本文件位于工程根目录下的build_local.sh,该脚本的主要功能有:

build_local.sh文件内容如下:

  1. #!/bin/sh
  2. #使用Gradle编译各个module
  3. ./gradlew clean
  4. ./gradlew build --stacktrace --debug
  5. #进入输出目录
  6. cd output
  7. #清空输出目录
  8. rm -rf *
  9. #创建输出子目录
  10. mkdir temp
  11. mkdir debug
  12. mkdir release
  13. #定义sdk版本号
  14. version="1.0.0"
  15. #定义模块是否打包标识
  16. is_include_hfasynchttp=true
  17. is_include_bitmapfun=true
  18. is_include_hfjson=true
  19. is_include_hflogger=true
  20. #省略其他...
  21. #解压所有debug版本的jar包到temp目录中
  22. cd temp
  23. if $is_include_hfasynchttp; then
  24. jar -xvf ../../hfasynchttp/build/intermediates/bundles/debug/classes.jar
  25. fi
  26. if $is_include_bitmapfun; then
  27. jar -xvf ../../hfbitmapfun/build/intermediates/bundles/debug/classes.jar
  28. fi
  29. if $is_include_hfjson; then
  30. jar -xvf ../../hfjson/build/intermediates/bundles/debug/classes.jar
  31. fi
  32. if $is_include_hflogger; then
  33. jar -xvf ../../hflogger/build/intermediates/bundles/debug/classes.jar
  34. fi
  35. #压缩所有debug版本的class文件到一个独立的jar包中
  36. jar -cvfM AndroidHyperion_${version}_debug.jar .
  37. #拷贝文件
  38. mv AndroidHyperion_${version}_debug.jar ../debug
  39. #清空temp目录
  40. rm -rf *
  41. #解压所有release版本的jar包到temp目录中
  42. if $is_include_hfasynchttp; then
  43. jar -xvf ../../hfasynchttp/build/intermediates/bundles/release/classes.jar
  44. fi
  45. if $is_include_bitmapfun; then
  46. jar -xvf ../../hfbitmapfun/build/intermediates/bundles/release/classes.jar
  47. fi
  48. if $is_include_hfjson; then
  49. jar -xvf ../../hfjson/build/intermediates/bundles/release/classes.jar
  50. fi
  51. if $is_include_hflogger; then
  52. jar -xvf ../../hflogger/build/intermediates/bundles/release/classes.jar
  53. fi
  54. #压缩所有release版本的class文件到一个jar包中
  55. jar -cvfM AndroidHyperion_${version}_release.jar .
  56. #拷贝文件
  57. mv AndroidHyperion_${version}_release.jar ../release
  58. #删除temp目录
  59. cd ..
  60. rm -rf temp

Jenkins构建

Jenkins编译脚本文件位于工程根目录下的build_jenkins.sh,该脚本的主要功能有:

可以看到,和本地构建唯一的区别就是分模块的参数化构建参数是定义在Jenkins上的,而不是定义在本地脚本中的,为了完整清晰起见,我们还是把完整的脚本文件贴出来:

  1. #!/bin/sh
  2. ./gradlew clean
  3. ./gradlew build --stacktrace --debug
  4. #进入输出目录
  5. cd output
  6. #清空输出目录
  7. rm -rf *
  8. #创建输出子目录
  9. mkdir temp
  10. mkdir debug
  11. mkdir release
  12. cd temp
  13. #解压所有debug版本的jar包
  14. if $is_include_hfasynchttp; then
  15. jar -xvf ../../hfasynchttp/build/intermediates/bundles/debug/classes.jar
  16. fi
  17. if $is_include_bitmapfun; then
  18. jar -xvf ../../hfbitmapfun/build/intermediates/bundles/debug/classes.jar
  19. fi
  20. if $is_include_hfjson; then
  21. jar -xvf ../../hfjson/build/intermediates/bundles/debug/classes.jar
  22. fi
  23. if $is_include_hflogger; then
  24. jar -xvf ../../hflogger/build/intermediates/bundles/debug/classes.jar
  25. fi
  26. #压缩所有debug版本的class文件到一个jar包中
  27. jar -cvfM AndroidHyperion_${version}_debug.jar .
  28. #移动生成的jar包到debug目录
  29. mv AndroidHyperion_${version}_debug.jar ../debug
  30. #清空temp目录
  31. rm -rf *
  32. #解压所有release版本的jar包
  33. if $is_include_hfasynchttp; then
  34. jar -xvf ../../hfasynchttp/build/intermediates/bundles/release/classes.jar
  35. fi
  36. if $is_include_bitmapfun; then
  37. jar -xvf ../../hfbitmapfun/build/intermediates/bundles/release/classes.jar
  38. fi
  39. if $is_include_hfjson; then
  40. jar -xvf ../../hfjson/build/intermediates/bundles/release/classes.jar
  41. fi
  42. if $is_include_hflogger; then
  43. jar -xvf ../../hflogger/build/intermediates/bundles/release/classes.jar
  44. fi
  45. #压缩所有release版本的class文件到一个jar包中
  46. jar -cvfM AndroidHyperion_${version}_release.jar .
  47. #移动生成的jar包到release目录
  48. mv AndroidHyperion_${version}_release.jar ../release
  49. #删除temp目录
  50. cd ..
  51. rm -rf temp

local和Jenkins参数化构建参数定义

类型 名称 默认值 描述
String version 1.0.0 Hyperion sdk版本号
Boolean is_include_hfasynchttp true 是否打包hfasynchttp
Boolean is_include_bitmapfun true 是否打包hfbitmapfun
Boolean is_include_hfjson true 是否打包hfjson
Boolean is_include_hflogger true 是否打包hflogger
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注