最近携程开源了一套动态加载的框架,总的来说,该框架和OpenAtlas还是有一定的相似之处的,比如资源的分区。此外该框架也支持热修复。个人觉得该框架中携程做的比较多的应该在打包语句的编写上面,这篇文章主要用于记录自己学习该框架的一个过程,携程的打包语句是我见过最复杂的,所以还是非常值得借鉴的。在携程的github上的DynamicAPK上,给出的打包方法是命令行执行gradle,如下
git clone https://github.com/CtripMobile/DynamicAPK.gitcd DynamicAPK/gradlew assembleRelease bundleRelease repackAll
该命令行中执行打包的语句gradlew assembleRelease bundleRelease repackAll,之后就会在对应目录下生成/build-outputs/appname-release-final.apk文件,这条打包语句可以分解为三条语句依次执行,即gradlew assembleRelease、gradlew bundleRelease、gradlew repackAll,我们依次来看这三个命令到底做了什么。
gradlew assembleRelease
该命令定义在sample模块的build.gradle文件中
//打包后产出物复制到build-outputs目录。apk、manifest、mappingtask copyReleaseOutputs(type:Copy){ from ("$buildDir/outputs/apk/sample-release.apk") { rename 'sample-release.apk', 'demo-base-release.apk' } from "$buildDir/intermediates/manifests/full/release/AndroidManifest.xml" from ("$buildDir/outputs/mapping/release/mapping.txt") { rename 'mapping.txt', 'demo-base-mapping.txt' } into new File(rootDir, 'build-outputs')}assembleRelease<<{ copyReleaseOutputs.execute()}
从上面的语句看到,在执行完assembleRelease的时候,还执行了copyReleaseOutputs这个task,而这个task所做的就是将sample目录下的build目录中生成的部分文件拷贝到build-outputs目录中
- 第一个文件是生成的apk文件,并对其进行了重命名;该文件用于后续插件打包的时候资源的引用等。
- 第二个文件是android的清单文件AndroidManifest.xml,直接复制不进行重命名;
- 第三个文件是mapping.txt文件,并对其进行了重名名。其中第三个文件是和代码混淆相关的,如果没有开启代码混淆,该文件是不存在的。
该task执行后,目录中生成的文件如图所示,其中mapping.txt文件的存在是因为我开启了混淆。
开启混淆的方式如下
buildTypes { ... release { ... minifyEnabled true ... }}
gradlew bundleRelease
之后执行的就是bundleRelease,这个task最终目的是生成插件so(后缀为so,本质还是apk,这也是很多加壳的应用反编译不出来什么东西的原因)
task bundleRelease (type:Zip,dependsOn:['compileRelease','aaptRelease','dexRelease']){ inputs.file "$buildDir/intermediates/dex/${project.name}_dex.zip" inputs.file "$buildDir/intermediates/res/resources.zip" outputs.file "${rootDir}/build-outputs/${apkName}.so" archiveName = "${apkName}.so" destinationDir = file("${rootDir}/build-outputs") duplicatesStrategy = 'fail' from zipTree("$buildDir/intermediates/dex/${project.name}_dex.zip") from zipTree("$buildDir/intermediates/res/resources.zip")}
该task会生成插件的相关so文件到build-outputs目录,该目录在会在其依赖的task中事先创建好,首先会在插件模块的build目录中将dex.zip和resources.zip压缩文件中的文件(这两个文件的生成在其依赖的task中完成)作为输入文件,重新压缩为一个so文件,so的名字为包名.so,其中包名中的点修改为了下划线,见下图
该task需要依赖其他三个Task,依次为aaptRelease、compileRelease、dexRelease
//初始化,确保必要目录都存在task init << { new File(rootDir, 'build-outputs').mkdirs() buildDir.mkdirs() new File(buildDir, 'gen/r').mkdirs() new File(buildDir, 'intermediates').mkdirs() new File(buildDir, 'intermediates/classes').mkdirs() new File(buildDir, 'intermediates/classes-obfuscated').mkdirs() new File(buildDir, 'intermediates/res').mkdirs() new File(buildDir, 'intermediates/dex').mkdirs()}task aaptRelease (type: Exec,dependsOn:'init'){ inputs.file "$sdk.androidJar" inputs.file "${rootDir}/build-outputs/demo-base-release.apk" inputs.file "$projectDir/AndroidManifest.xml" inputs.dir "$projectDir/res" inputs.dir "$projectDir/assets" inputs.file "${rootDir}/sample/build/generated/source/r/release/ctrip/android/sample/R.java" outputs.dir "$buildDir/gen/r" outputs.file "$buildDir/intermediates/res/resources.zip" outputs.file "$buildDir/intermediates/res/aapt-rules.txt" workingDir buildDir executable sdk.aapt def resourceId='' def parseApkXml=(new XmlParser()).parse(new File(rootDir,'apk_module_config.xml')) parseApkXml.Module.each{ module-> if( module.@packageName=="${packageName}") { resourceId=module.@resourceId println "find packageName: " + module.@packageName + " ,resourceId:" + resourceId } } def argv = [] argv << 'package' //打包 argv << "-v" argv << '-f' //强制覆盖已有文件 argv << "-I" argv << "$sdk.androidJar" //添加一个已有的固化jar包 argv << '-I' argv << "${rootDir}/build-outputs/demo-base-release.apk" argv << '-M' argv << "$projectDir/AndroidManifest.xml" //指定manifest文件 argv << '-S' argv << "$projectDir/res" //res目录 argv << '-A' argv << "$projectDir/assets" //assets目录 argv << '-m' //make package directories under location specified by -J argv << '-J' argv << "$buildDir/gen/r" //哪里输出R.java定义 argv << '-F' argv << "$buildDir/intermediates/res/resources.zip" //指定apk的输出位置 argv << '-G' //-G A file to output proguard options into. argv << "$buildDir/intermediates/res/aapt-rules.txt" // argv << '--debug-mode' //manifest的application元素添加android:debuggable="true" argv << '--custom-package' //指定R.java生成的package包名 argv << "${packageName}" argv << '-0' //指定哪些后缀名不会被压缩 argv << 'apk' argv << '--public-R-path' argv << "${rootDir}/sample/build/generated/source/r/release/ctrip/android/sample/R.java" argv << '--apk-module' argv << "$resourceId" args = argv}
可以看到输出了一个resources.zip文件,这个文件就是bundleRelease 中用到的压缩文件之一,总的来说该task就是拼接命令行参数生成文件。
aaptRelease是对插件资源文件的编译,依赖于aapt命令行工具,在了解该Task之前,需要了解一下该命令的一些参数。
- -I add an existing package to base include set
这个参数可以在依赖路径中追加一个已经存在的package。在Android中,资源的编译也需要依赖,最常用的依赖就是SDK自带的android.jar本身。打开android.jar可以看到,其实不是一个普通的jar包,其中不但包含了已有SDK类库class,还包含了SDK自带的已编译资源以及资源索引表resources.arsc文件。在日常的开发中,[email protected]:[email protected]??于编译过程中aapt对android.jar的依赖引用。同理,我们也可以使用这个参数引用一个已存在的apk包作为依赖资源参与编译。
- -G A file to output proguard options into.
资源编译中,对组件的类名、方法引用会导致运行期反射调用,所以这一类符号量是不能在代码混淆阶段被混淆或者被裁减掉的,否则等到运行时会找不到布局文件中引用到的类和方法。-G方法会导出在资源编译过程中发现的必须keep的类和接口,它将作为追加配置文件参与到后期的混淆阶段中。
- -J specify where to output R.java resource constant definitions
在Android中,所有资源会在Java源码层面生成对应的常量ID,这些ID会记录到R.java文件中,参与到之后的代码编译阶段中。在R.java文件中,Android资源在编译过程中会生成所有资源的ID,作为常量统一存放在R类中供其他代码引用。在R类中生成的每一个int型四字节资源ID,实际上都由三个字段组成。第一字节代表了Package,第二字节为分类,三四字节为类内ID。
在对插件的编译过程中,携程主要用了三个参数。其中也不乏携程自己改装aapt增加的参数。如下
- 使用-I参数对宿主的apk进行引用。
据此,插件的资源、xml布局中就可以使用宿主的资源和控件、布局类了。
- 为aapt增加–apk-module参数。
资源ID其实有一个PackageID的内部字段。我们为每个插件工程指定独特的PackageID字段,这样根据资源ID就很容易判明,此资源需要从哪个插件apk中去查找并加载了。
- 为aapt增加–public-R-path参数。
按照对android.jar包中资源使用的常规手段,引用系统资源可使用它的R类的全限定名android.R来引用具体ID,以便和当前项目中的R类区分。插件对于宿主的资源引用,当然也可以使用base.package.name.R来完成。但由于历史原因,各子BU的“插件”代码是从主app中解耦独立出去的,资源引用还是直接使用当前工程的R。如果改为标准模式,则当前大量遗留代码中R都需要酌情改为base.R,工程量大并且容易出错,未来对bu开发人员的使用也有点不够“透明”。因此我们在设计上做了让步,额外增加–public-R-path参数,为aapt指明了base.R的位置,让它在编译期间把base的资源ID定义在插件的R类中完整复制一份,这样插件工程即可和之前一样,完全不用在乎资源来自于宿主或者自身,直接使用即可。当然这样做带来的副作用就是宿主和插件的资源不应有重名,这点我们通过开发规范来约束,相对比较容易理解一些。
了解了这么一些基础的概念之后,回头再来看看该task所做的工作。首先调用了task init进行一些目录的创建,然后引入创建apk资源文件所有必要的文件,再通过检查apk_module_config.xml文件,找到对应包名的resourceId,该文件的定义如下
<?xml version="1.0" encoding="utf-8"?><ApkModules> <Module packageName="ctrip.android.demo1" resourceId="0x31"/> <Module packageName="ctrip.android.demo2" resourceId="0x36"/></ApkModules>
之后做的就是拼接命令行语句,执行生成资源就可以了。而拼接的命令行语句中,指定了很多参数,如-I、–apk-module、–public-R-path等等,具体意义在上文已经解释过了,最终的产物就是资源文件的压缩包resources.zip。
compileRelease这个task的作用就是编译java文件,会指定classpath目录以及目标目录等相关信息。
task compileRelease(type: JavaCompile,dependsOn:'aaptRelease') { inputs.file "$sdk.androidJar" inputs.files fileTree("${projectDir}/libs").include('*.jar') inputs.file "${rootDir}/sample/build/intermediates/classes-proguard/release/classes.jar" inputs.files fileTree("$projectDir/src").include('**/*.java') inputs.files fileTree("$buildDir/gen/r").include('**/*.java') outputs.dir "$buildDir/intermediates/classes" sourceCompatibility = '1.6' targetCompatibility = '1.6' classpath = files( "${sdk.androidJar}", "${sdk.apacheJar}", fileTree("${projectDir}/libs").include('*.jar'), "${rootDir}/sample/build/intermediates/classes-proguard/release/classes.jar" ) destinationDir = file("$buildDir/intermediates/classes") dependencyCacheDir = file("${buildDir}/dependency-cache") source = files(fileTree("$projectDir/src").include('**/*.java'), fileTree("$buildDir/gen/r").include('**/*.java')) options.encoding = 'UTF-8'}
最终的生成文件会在build/intermediates/classes中,可以看出最终的产物应该是一些列的class类文件
dexRelease这个task的作用就是根据compileRelease生成的classes文件,调用dx命令行工具打包成android专用的dex文件。
task dexRelease (type:Exec){ inputs.file "${buildDir}/intermediates/classes" outputs.file "${buildDir}/intermediates/dex/${project.name}_dex.zip" workingDir buildDir executable sdk.dex def argv = [] argv << '--dex' argv << "--output=${buildDir}/intermediates/dex/${project.name}_dex.zip" argv << "${buildDir}/intermediates/classes" args = argv}
这个task输出了一个dex.zip,也是bundleRelease这个task中用到的一个压缩包之一。
gradlew repackAll
这个task主要是调用了其他5个task
task repackAll(dependsOn: ['reload','resign','repack','realign','concatMappings'])
下面来一一分析这几个task
reload的作用就是往最开始生成的宿主文件的apk的assets目录中,添加插件so,而so正是前面几个task生成的插件so文件,最终的产物是demo-release-reloaded.apk这个文件
//base apk的assets中填充各子apk//输入:Ctrip-base-release.apk//输出:Ctrip-release-reloaded.apktask reload(type:Zip){ inputs.file "$rootDir/build-outputs/demo-base-release.apk" inputs.files fileTree(new File(rootDir,'build-outputs')).include('*.so') outputs.file "$rootDir/build-outputs/demo-release-reloaded.apk" into 'assets/baseres/',{ from fileTree(new File(rootDir,'build-outputs')).include('*.so') } from zipTree("$rootDir/build-outputs/demo-base-release.apk"), { exclude('**/META-INF/*.SF') exclude('**/META-INF/*.RSA') } destinationDir file("$rootDir/build-outputs/") archiveName 'demo-release-reloaded.apk'}
apk文件发生了改变,需要对其进行重新签名,resign这个task的目的就是这个,调用命令行签名工具,添加证书的信息进行签名,但是在签名前会进行一次压缩,repack 这个task就是进行这个操作,最后输出的是demo-release-repacked.apk,打包完毕后便会进行签名的操作,也就是resign这个task所做的工作
//对apk重新压缩,调整各文件压缩比到正确//输入:Ctrip-release-reloaded.apk//输出:Ctrip-release-repacked.apktask repack (dependsOn: 'reload') { inputs.file "$rootDir/build-outputs/demo-release-reloaded.apk" outputs.file "$rootDir/build-outputs/demo-release-repacked.apk" doLast{ println "release打包之后,重新压缩一遍,以压缩resources.arsc" def oldApkFile = file("$rootDir/build-outputs/demo-release-reloaded.apk") assert oldApkFile != null : "没有找到release包!" def newApkFile = new File(oldApkFile.parentFile, 'demo-release-repacked.apk') //重新打包 repackApk(oldApkFile.absolutePath, newApkFile.absolutePath) assert newApkFile.exists() : "没有找到重新压缩的release包!" }}
//对apk重签名//输入:Ctrip-release-repacked.apk//输出:Ctrip-release-resigned.apktask resign(type:Exec,dependsOn: 'repack'){ inputs.file "$rootDir/build-outputs/demo-release-repacked.apk" outputs.file "$rootDir/build-outputs/demo-release-resigned.apk" workingDir "$rootDir/build-outputs" executable "${System.env.'JAVA_HOME'}/bin/jarsigner" def argv = [] argv << '-verbose' argv << '-sigalg' argv << 'SHA1withRSA' argv << '-digestalg' argv << 'SHA1' argv << '-keystore' argv << "$rootDir/demo.jks" argv << '-storepass' argv << '123456' argv << '-keypass' argv << '123456' argv << '-signedjar' argv << 'demo-release-resigned.apk' argv << 'demo-release-repacked.apk' argv << 'demo' args = argv}
签名完毕后会输出签名后的文件demo-release-resigned.apk
而repack这个task最终调用的是repackApk重新进行压缩打包的
import java.util.zip.ZipEntryimport java.util.zip.ZipFileimport java.util.zip.ZipOutputStream// 打包过程中很多手工zip过程:// 1,为了压缩resources.arsc文件而对标准产出包重新压缩// 2,以及各子apk的纯手打apk包// 但对于音频等文件,压缩会导致资源加载报异常// 重新打包方法,使用STORED过滤掉不应该压缩的文件们// 后缀名列表来自于android源码def repackApk(originApk, targetApk){ def noCompressExt = [".jpg", ".jpeg", ".png", ".gif", ".wav", ".mp2", ".mp3", ".ogg", ".aac", ".mpg", ".mpeg", ".mid", ".midi", ".smf", ".jet", ".rtttl", ".imy", ".xmf", ".mp4", ".m4a", ".m4v", ".3gp", ".3gpp", ".3g2", ".3gpp2", ".amr", ".awb", ".wma", ".wmv"] ZipFile zipFile = new ZipFile(originApk) ZipOutputStream zos = new ZipOutputStream(new BufferedOutputStream(new FileOutputStream(targetApk))) zipFile.entries().each{ entryIn -> if(entryIn.directory){ println "${entryIn.name} is a directory" } else{ def entryOut = new ZipEntry(entryIn.name) def dotPos = entryIn.name.lastIndexOf('.') def ext = (dotPos >= 0) ? entryIn.name.substring(dotPos) : "" def isRes = entryIn.name.startsWith('res/') if(isRes && ext in noCompressExt){ entryOut.method = ZipEntry.STORED entryOut.size = entryIn.size entryOut.compressedSize = entryIn.size entryOut.crc = entryIn.crc } else{ entryOut.method = ZipEntry.DEFLATED } zos.putNextEntry(entryOut) zos << zipFile.getInputStream(entryIn) zos.closeEntry() } } zos.finish() zos.close() zipFile.close()}
当然,签名完毕后会对该apk进行4K对齐操作。
//重新对jar包做对齐操作//输入:Ctrip-release-resigned.apk//输出:Ctrip-release-final.apktask realign (dependsOn: 'resign') { inputs.file "$rootDir/build-outputs/demo-release-resigned.apk" outputs.file "$rootDir/build-outputs/demo-release-final.apk" doLast{ println '重新zipalign,还可以加大压缩率!' def oldApkFile = file("$rootDir/build-outputs/demo-release-resigned.apk") assert oldApkFile != null : "没有找到release包!" def newApkFile = new File(oldApkFile.parentFile,'demo-release-final.apk') def cmdZipAlign = getZipAlignPath() def argv = [] argv << '-f' //overwrite existing outfile.zip // argv << '-z' //recompress using Zopfli argv << '-v' //verbose output argv << '4' //alignment in bytes, e.g. '4' provides 32-bit alignment argv << oldApkFile.absolutePath argv << newApkFile.absolutePath project.exec { commandLine cmdZipAlign args argv } assert newApkFile.exists() : "没有找到重新zipalign的release包!" }}
最后还有一个task,就是concatMappings,这个task很简单,做的就是合并一下mapping文件。
/** * 用来连接文件的task */class ConcatFiles extends DefaultTask { @InputFiles FileCollection sources @OutputFile File target @TaskAction void concat() { File tmp = File.createTempFile('concat', null, target.getParentFile()) tmp.withWriter { writer -> sources.each { file -> file.withReader { reader -> writer << reader } } } target.delete() tmp.renameTo(target) }}//合并base和所有模块的mapping文件task concatMappings(type: ConcatFiles){ sources = fileTree(new File(rootDir,'build-outputs')).include('*mapping.txt') target = new File(rootDir,'build-outputs/demo-mapping-final.txt')}
最终repackAll这个task的产物如下
以上就是携程动态加载框架的打包流程分析,纯属个人看法,如有不正确的地方,请给予指正。
版权声明:本文为博主原创文章,未经博主允许不得转载。
- 3楼jjack7昨天 14:10
- 博主方便加下qq吗,我遇到了一些别的问题。。谢谢
- 2楼jjack7昨天 13:47
- 发现这样修改后可以启动打包了n repositories {n jcenter()n //maven { url "http://mirrors.ibiblio.org/maven2"}(之前注释的是jcenter())n }n但是到最后还是有错:n* What went wrong:nExecution failed for task ':sample:resign'.n> A problem occurred starting process 'command 'null/bin/jarsigner''
- 1楼jjack7昨天 11:24
- 博主,我下了demo后跑不起来!报java.io.FileNotFoundException: /data/data/ctrip.android.sample/files/storage/meta: open failed: ENOENT (No such file or directory)n求解答!
- Re: sbsujjbcy昨天 11:38
- 回复jjack7n使用gradlew assembleRelease bundleRelease repackAll进行打包,对应的apk在build-outputs中
- Re: jjack7昨天 12:54
- 回复sbsujjbcyn后面还有但是回复里贴不出来了,gradle我一种用1.3.0,平时都是正常的,怎么现在不行了呢?
- Re: jjack7昨天 12:58
- 回复sbsujjbcyn我这样做了但是遇到了这个:nFAILURE: Build failed with an exception.nn* What went wrong:nA problem occurred configuring root project 'DynamicAPK-master'.n> Could not resolve all dependencies for configuration ':classpath'.n > Could not find com.android.tools.build:gradle:1.3.0.n Searched in the following locations:nhttp://mirrors.ibiblio.org/maven2/com/android/tools/build/gradle/1.3.0/gradle-1.3.0.pom