技术标签: android+打包+不同app
0x01 基本项目结构
使用Android Studio创建的Android项目会划分成三个层级:
project : settings.gradle定义了构建应用时包含了哪些模块;build.gradle定义了适用于项目中所有模块的构建配置
module : 可以是一个app类型的module,对应生成apk应用;也可以是一个lib类型的module,对应生成aar包. 每个module中包含的build.gradle定义了针对该module的各种构建选项
sourceset : 每个module的源码和资源分组为多个源集,main源集包含了所有变体共用的源码和资源. 源集结合productFlavors和buildTypes编译可以很好的实现多套源码,资源,以及依赖库的应用打包
+--- Comi/
|
\--- build.gradle
\--- setting.gradle
+--- app/
| |
| \--- build.gradle
| \--- build/
| \--- libs/
| \--- src/
| |
| \--- main/
| | \--- java/
| | \--- com.icomico.comi/
| | \--- res/
| | \--- drawable/
| | \--- layout/
| | \--- ...
| | \--- AndroidManifest.xml
| \--- jp/
| \--- ...
+--- comi_base/
+--- ...
0x02 构建流程
构建流程涉及到了将项目转化成apk包的各方面,并配合有可灵活配置的构建选项,基本的构建流程包含了以下几步:
编译器将源码转换成dex文件,并将其它所有内容转换成已编译资源
打包工具将dex文件和已编译资源合并成单个apk文件,apk文件此时处于未签名状态
打包工具根据当前构建任务对应buildTypes中选择的signingConfigs选项,使用相应秘钥对apk文件对apk进行签名
打包工具使用zipalign工具对apk包进行优化
构建结束,生成最终apk文件
basic build process
0x03 灵活配置构建选项
基于Gradle的Android插件提供了一系列可自定义配置的构建选项,可以方便在构建过程中选择多样的构建方式,以下列出一些常用的构建选项
defaultConfig
包含了用于所有变体种的一些构建选项,如果productFlavors中配置了同样的构建选项,则会覆盖defaultConfig中的对应项
defaultConfig {
applicationId 'com.icomico.comi' //应用包名
minSdkVersion 10
targetSdkVersion 23
versionCode 1000
versionName '1.0'
multiDexEnabled true //支持多dex文件,用于解决方法数超过65K的限制时打包输出问题
}
signingConfigs
包含了app签名文件的各项配置
signingConfigs {
release {
storeFile file('xxx')
storePassword 'xxx'
keyAlias 'xxx'
keyPassword 'xxx'
v2SigningEnabled false //是否使用7.0版本引入的APK signature scheme v2签名方式,默认为true
}
debug {
...
}
}
buildTypes
构建类型中可定义多个构建类型,每个构建类型可以选择不同的构建属性,包括使用的签名配置,优化选项,混淆选项等.必须定义至少一个构建类型才能开始构建应用
buildTypes {
release {
signingConfig signingConfigs.release //使用的签名配置
minifyEnabled true //是否使用proguard进行代码压缩优化
shrinkResources true //是否进行资源压缩,使用资源压缩前需要使用代码压缩
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro', 'proguard-fresco.pro' //指定使用的proguard配置文件
manifestPlaceholders = [DEBUG: false] //替换manifest文件中标记为${DEBUG}的文本
}
debug {
...
}
}
productFlavors
谷歌翻译为产品风味的选项,用来发布不同的应用版本,并可以为不同版本指定包括包名,版本号等不同的构建属性
productFlavors {
develop {
applicationId 'com.icomico.comi.dev' //应用包名
manifestPlaceholders = [TEMP_CH_NAME name]} //使用flavor名称替换manifest中的渠道号字段,实现修改渠道号
official {
applicationId 'com.icomico.comi'
manifestPlaceholders = [TEMP_CH_NAME: name]
}
...
}
sourceSets
源集可指定了不同变体使用的源码和资源文件,默认情况下,源集与src目录下的目录结构一一对应,每个src下的目录为一个源集.sourceSets配置项可在变体基础上指定变体使用的源集文件位置,这样可以在定义了大量变体,同时使用的源码和资源文件没有区别的情况下,避免在src下生成同等数量的目录
sourceSets {
develop {
jni.srcDir 'src/main/jni'
java.srcDir 'src/dev/java'
res.srcDir 'src/dev/res'
manifest.srcFile 'src/dev/AndroidManifest.xml'
}
official {
...
}
...
}
dependencies
依赖项用以管理来自本地或远程地址上的项目依赖,包括了模块依赖,远程和本地的库文件依赖. 同时可以指定使用依赖项的方式以及为不同变体指定不同的依赖项
dependencies {
compile project(':comi_base') //编译时依赖,gradle 将此配置的依赖项添加到类路径和应用的apk
apk files('libs/apk_use.jar') //仅运行时依赖,需要将其与apk一起打包.只可用于jar包形式的依赖项
provided files('libs/apk_unuse.jar') //不与apk一起打包的编译时依赖,只可用于jar包形式的依赖项
officialCompile project(':comi_player') //构建official版本时使用的依赖项
debugCompile 'com.squareup.leakcanary:leakcanary-android:1.3' //构建debug编译类型时使用的依赖项
releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.3' //构建release编译类型时使用的依赖项
baiduCompile files('libs/BDAutoUpdateSDK_20150605_V1.2.0.jar') //构建baidu渠道版本时使用的依赖项
baiduCompile files('libs/need_lib.jar')
baiduCompile files('libs/patchupdate.jar')
}
0x04 多种变体的打包实践
基于module划分的项目结构
利用module将项目工程划分为多个app module和多个lib module的结构,多个app工程可以各自选择需要使用到的lib工程,每个lib工程实现相对独立的功能模块.
+--- Comi/
|
+--- app/
| |
| \--- build.gradle
| \--- src/
| |
| \--- main/
| | \--- java/
| | \--- com.icomico.comi/
| | \--- res/
| \--- jp/
| | \--- java/
| | \--- com.icomico.comi/
| | |
| | \--- ChConfig.java
| \--- official/
| | \--- java/
| | \--- com.icomico.comi/
| | |
| | \--- ChConfig.java
| \--- ...
| \ ...
+--- app_special/
+--- comi_base/
+--- comi_reader/
+--- comi_player/
+--- comi_web/
+--- comi_danmaku/
+--- ...
使用build.gradle脚本管理module的构建配置
android {
defaultConfig { ... }
buildTypes { ... }
signingConfigs { ... }
sourceSets { ... }
productFlavors {
develop {
applicationId 'com.icomico.comi.dev' //应用包名
manifestPlaceholders = [TEMP_CH_NAME name]} //使用flavor名称替换manifest中的渠道号字段,实现修改渠道号
official {
applicationId 'com.icomico.comi'
manifestPlaceholders = [TEMP_CH_NAME: name]
}
...
}
...
}
dependencies {
compile project(':comi_base')
officialCompile project(':comi_player')
...
}
变体
由buildTypes, productFlavors, sourceSets三项配置, 可以形成gradle打包任务中的多种变体, gradle针对每种变体组合都创建了一个构建任务assemble,可指定执行不同变体的构建任务
为变体指定包名,替换指定字段,指定依赖项等
在app工程的build.gradle构建脚本中,使用productFlavors来定义了多个渠道的变体,并为不同渠道设置了不同的应用包名,渠道号等.同时可以在dependencies中为不同的变体选择了依赖的lib工程,如comi_player库仅为official渠道的变体集成
不同变体使用多套代码及资源
针对代码,在app工程的src目录下,利用sourceset划分出official和jp两个源集,其中各自定义了用一份java类ChConfig.java,在打包中不同源集的apk文件包含的便是不同的代码文件.main目录下的代码文件会包含在每个源集中,因此在main目录下就不能定义同样的ChConfig.java文件
针对资源,与代码文件相同,可以在源集目录下定义同名的资源,生成apk包时会覆盖main目录下的同名资源
0x05 渠道打包
结合持续集成环境,可以将渠道号等参数以一定规则作为文件名,在已生成好的apk文件中的META-INF目录下写入空文件,以提升渠道包打包速度
0x06 参考文献
文章浏览阅读1.1k次。2020版本,也许其它安装类型的脚本也可以用同样的方式清理删除""文件夹,注意里面是配置文件,删除前记得备份。同时清理注册表。_mmd文件导入3ds max
文章浏览阅读4.6k次,点赞2次,收藏19次。1、第一个登录页面,里面有提交表单,action提交到index页面2、第二个页面,可以使用第一个页面的参数,实现一个数据不同页面之间的传递效果3、第二个页面之所以可以使用第一个页面的数据,就是利用了URL里面的location.search参数4、第二个页面提取参数5、去掉?6、分隔符分开表单form:可以将内容提交到另外一个页面上输入页面:<!DOCTYPE html><html lang="en"><head> ._cshtml 跳转页面并传整行数据
文章浏览阅读829次,点赞2次,收藏17次。许愿墙 许下你的愿望_编程实现许愿墙的页面结构,效果图如下:
文章浏览阅读1.9w次,点赞2次,收藏3次。A公司准备对他下面的N个产品评选最差奖,评选的方式是首先对每个产品进行评分,然后根据评分区间计算相邻几个产品中最差的产品。评选的标准是依次找到从当前产品开始前M个产品中最差的产品,请给出最差产品的评分序列。_最差产品奖
文章浏览阅读922次,点赞10次,收藏5次。Windows直接运行python程序_windows 系统跑python程序
文章浏览阅读1.2k次,点赞3次,收藏3次。R语言读取(加载)txt格式数据为dataframe_r语言read.dataframe
文章浏览阅读1.3k次。首先,问题是如何出现的?晚上复查代码,发现一个activity没有调用自己的ondestroy方法我表示非常的费解,于是我检查了下代码。发现再finish代码之后接了如下代码finish();System.exit(0);//这就是罪魁祸首为什么这样写会出现问题System.exit(0);////看一下函数的原型public static void exit (int code)//Added ..._android 手动杀死app,activity不执行ondestroy
文章浏览阅读894次。Q: SylixOS 版权是什么形式, 是否分为<开发版税>和<运行时版税>.A: SylixOS 是开源并免费的操作系统, 支持 BSD/GPL 协议(GPL 版本暂未确定). 没有任何的运行时版税. 您可以用她来做任何 您喜欢做的项目. 也可以修改 SylixOS 的源代码, 不需要支付任何费用. 当然笔者希望您可以将使用 SylixOS 开发的项目 (不需要开源)或对 SylixOS 源码的修改及时告知笔者.需要指出: SylixOS 本身仅是笔者用来提升自己水平而开发的_select函数 导致堆栈溢出 sylixos
文章浏览阅读135次。我们要通过实现三个类,来实现一个web后端的全局异常管理功能,捕获controller层异常,封装成map集合,进行返回1.实现异常处理的工具类实现一个imooc.naga.core.exception,异常处理的包,定义两个类分别是:1.系统返回状态标识码实体类,2:封装了异常状态值,和异常信息内容的实体类1.系统返回状态标识码实体类package imooc.naga.core.exception;/** * 定义我们系统返回值的状态标志 */public class ErrorCo_naga数据分析平台
文章浏览阅读394次,点赞8次,收藏9次。fileFolder=fullfile('G:\气象数据\sitedata\day_temperature1951_2017');%将从该文件夹中提取的数据并入到上一个文件夹中的数据,;%如果该文件夹里有该站点的数据,则min_line不是空。_读取气象a文件
文章浏览阅读2.2k次。2016年0..._c#获取此电脑的便携设备信息
文章浏览阅读2.2k次,点赞5次,收藏9次。乐观锁是一种并发控制的策略,它假设多个线程在操作共享数据时,不会发生冲突,因此不需要加锁,而是在更新数据时,通过比较当前状态和上一次的状态,来判断是否有其他线程修改了数据。版本号控制的思想是,在数据表中增加一个版本号字段(或者时间戳字段),每次更新数据时,都将版本号加一(或者更新时间戳),并且在更新时,检查当前的版本号是否和数据库中的一致,如果一致,就执行更新操作;ABA问题是指,在一个线程执行CAS操作时,发现内存中的值没有变化(仍然为A),就认为没有其他线程修改过这个值,然后执行更新操作。_乐观锁