Android Studio多渠道打包及自动化构建
创始人
2024-05-25 22:47:39
0

Android 有不同的应用市场,也就是不同的渠道,需要为每个应用市场打一个安装包,但主要的代码是一样的,可能部分资源不一样,部分代码不一样,如果每个渠道都需要修改,然后打包,非常耗时。所以 AS 是提供了多渠道打包的。

可能遇到的需求

  1. 不同渠道 applicationId 不一样;
  2. 不同渠道配置参数不一样;
  3. 不同渠道签名文件不一样;
  4. 不同渠道资源文件不一样;
  5. 不同渠道部分代码不一样;
  6. 不同渠道依赖不一样;

这里会先说一下初级版配置,再说升级版配置—— Grovvy 进行自动化构建。

初级版多渠道配置

productFlavors :不同产品口味,就是AS自带的不同渠道打包关键字。可以进行多渠道配置,有两种方式。

1、在 app 模块下的 build.gradle 配置

// 读取不同的签名文件
def getSignProperties(filename){File signConfigFile = file("${rootProject.rootDir}/app/keystore/${filename}.properties")Properties signProperties = new Properties()signProperties.load(new FileInputStream(signConfigFile))return signProperties
}android {compileSdk 32defaultConfig {applicationId "com.XXX"minSdk 21targetSdk 32versionCode 5versionName "3.0"testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"}// 不同渠道的签名
signingConfigs {release {def signProperties = getSignProperties('signing')storeFile file(signProperties['KEYSTORE_FILE'])storePassword signProperties['KEY_PASSWORD']keyAlias signProperties['KEY_ALIAS']keyPassword signProperties['KEY_PASSWORD']}//不同的渠道,定义不同的签名文件huawei {def signProperties = getSignProperties('signing-huawei')storeFile file(signProperties['KEYSTORE_FILE'])storePassword signProperties['KEY_PASSWORD']keyAlias signProperties['KEY_ALIAS']keyPassword signProperties['KEY_PASSWORD']}xiaomi {def signProperties = getSignProperties('signing-xiaomi')storeFile file(signProperties['KEYSTORE_FILE'])storePassword signProperties['KEY_PASSWORD']keyAlias signProperties['KEY_ALIAS']keyPassword signProperties['KEY_PASSWORD']}}// 配置不同渠道参数productFlavors{huawei{applicationId ="com.xxx"//渠道参数buildConfigField "String", "token", "\"XXXX\""// manifest 读取的参数,在 manifest 里如何使用,见后文manifestPlaceholders=["app_name":"CCCCC"]}// 其他渠道类似}	// 配置打包签名buildTypes {debug {minifyEnabled falsedebuggable trueshrinkResources falseproguardFiles getDefaultProguardFile('proguard-android-optimize.txt'),'proguard-rules.pro'signingConfig signingConfigs.release}release {minifyEnabled truedebuggable falseshrinkResources falseproguardFiles getDefaultProguardFile('proguard-android-optimize.txt'),'proguard-rules.pro'//signingConfig signingConfigs.releaseproductFlavors.xiaoxing236.signingConfig signingConfigs.huaweiproductFlavors.xiaoxing238.signingConfig signingConfigs.xiaomi}}// 指定打包输出的路径applicationVariants.all { variant ->// 打包完成后输出路径def name = variant.flavorName +"_" + variant.buildType.name +"_" + variant.versionName +"_" + new Date().format('yyyyMMddhhmm') + ".apk"//相对路径app/build/outputs/apk/huawei/release/def path = "../../../../../apk/" //相当于路径 app/apk/variant.outputs.each { output ->def outputFile = output.outputFileif (outputFile != null && outputFile.name.endsWith('.apk') && outputFile.name.contains('release')) {//指定路径输出output.outputFileName = new File(path, name)}}}//不同渠道不同资源文件// sourceSets{ } 源文件目录设置sourceSets {// 公共代码及资源main {jniLibs.srcDirs = ['libs']}// 不同资源huawei.res.srcDirs 'src/huawei/res'xiaomi.res.srcDirs 'src/xiaomi/res'// 其他渠道类似,以下不再重复//不同代码huawei.java.srcDirs 'src/huawei/java'xiaomi.java.srcDirs 'src/xiaomi/java'// 不同渠道 manifest 文件huawei.manifest.srcFile 'src/huawei/AndroidManifest.xml'xiaomi.manifest.srcFile 'src/xiaomi/AndroidManifest.xml'}}// 不同渠道的依赖
dependencies {// 公共的依赖implementation 'ccccc'// 不同渠道依赖xiaomiApi('xxxxxxx')huaweiImplementation('xxxxxxxx')
}

不同渠道配置的参数需要在 manifest 里使用




在代码里使用 buildconfig 参数

private final String TOKEN = BuildConfig.token;

2、可以把以上不同渠道的配置单独放在一个 flavor.gradle 文件里,该文件与 setting.gradle 目录同级。 然后在 app 模块的 build.gradle 引用 flavor.gradle 文件即可。

apply from: ("${rootProject.rootDir}/flavor.gradle")

升级版渠道配置

按照以上配置方式,每增加一个渠道,就得每个渠道重新写一遍 huawei.manifest.srcFile 等这种操作,会让 build.gradle 显得非常臃肿。可以通过固定规则,写脚本解决以上问题。

1、在项目中创建出打包脚本文件夹 buildSrc,在此文件夹下创建 src/resource/**META-INF/gradle-plugins 路径及文件夹名固定。**

在这里插入图片描述

2、定义自动构建插件路径,在 src/resource/**META-INF/gradle-plugins 路径下创建一个 xxx.properties 文件,文件内定义构建脚本路径。**

// 路径是写脚本的文件路径
implementation-class=com.xxx.plugin.PackagePlugin

3、在 build.gradle 里引入相关仓库

//依赖 groovy 插件,这个是 Gradle 内置的插件
plugins {`kotlin-dsl``java-gradle-plugin`groovy
}val androidGradlePlugin = "com.android.tools.build:gradle:4.2.2"
val kotlin_version = "1.6.10"//引入相关的仓库
dependencies {// 导入androidGradlePlugin,这样buildSrc可以使用gradle相关apiimplementation(androidGradlePlugin)// Depend on the kotlin plugin, since we want to access it in our pluginimplementation("org.jetbrains.kotlin:kotlin-gradle-plugin:${kotlin_version}")// Depend on the default Gradle API since we want to build a custom pluginimplementation(gradleApi())implementation(localGroovy())
}

4、在主模块(app)模块的 build.gradle 中引入插件。

// plugin 的名字是第 2 步创建 properties 的名字
apply plugin: 'PackPlugin'

5、在 PackagePlugin 中开始编写自动构建脚本

编写脚本用的是 groovy 语法,可以参考这篇文章:Gradle插件从入门到进阶

class PackagePlugin : Plugin {// plugin 必须实现的方法override fun apply(target: Project) {// 获取 android extensionvar appExtension = target.extensions.getByName("android") as AppExtension// 多渠道构建appExtension.productFlavors {var channelList = getChannelList()channelList.forEach { channelModel ->register(channelModel.channelName) {// 每个渠道的需要配置的参数,可以根据自己的规则订applicationId = channelModel.packageNameversionCode = channelModel.versionCodeversionName = "${channelModel.versionCode}.0"// manifest 需要配置的参数manifestPlaceholders["ads_id"] = channelModel.adsIdmanifestPlaceholders["app_name"] = channelModel.appName// 代码里需要使用的不同渠道配置参数buildConfigField("String", "XXX", "\"${channelModel.定义的属性名}\"")buildConfigField("String", "XXX", "\"${channelModel.定义的属性名}\"")}}}// 签名文件appExtension.signingConfigs {var channelList = getChannelList()channelList.forEach { channelModel ->var channelName = channelModel.channelNameregister(channelName) {// 可以单独处理不一样的包storeFile(getKeyStoreFile(channelName, target))storePassword(channelName)keyAlias(channelName)keyPassword(channelName)}}}// 不同渠道配置不同的签名文件,签名文件的名字、别名、密码可以自行定义appExtension.signingConfigs.forEach { signingConfig ->println("PackagePlugin signing:${signingConfig.keyAlias.toString()}")appExtension.productFlavors.getByName(signingConfig.keyAlias.toString()).signingConfig =signingConfig}// 不同渠道的不同代码、资源、和 manifestappExtension.sourceSets {var channelList = getChannelList()channelList.forEach { channelModel ->var channelName = channelModel.channelNamegetByName(channelName) {res.srcDirs("src/${channelName}/res")java.srcDirs("src/${channelName}/java")manifest.srcFile("src/${channelName}/AndroidManifest.xml")}}}}}

相关内容

热门资讯

Python|位运算|数组|动... 目录 1、只出现一次的数字(位运算,数组) 示例 选项代...
张岱的人物生平 张岱的人物生平张岱(414年-484年),字景山,吴郡吴县(今江苏苏州)人。南朝齐大臣。祖父张敞,东...
西游西后传演员女人物 西游西后传演员女人物西游西后传演员女人物 孙悟空 六小龄童 唐僧 徐少华 ...
名人故事中贾岛作诗内容简介 名人故事中贾岛作诗内容简介有一次,贾岛骑驴闯了官道.他正琢磨着一句诗,名叫《题李凝幽居》全诗如下:闲...
和男朋友一起优秀的文案? 和男朋友一起优秀的文案?1.希望是惟一所有的人都共同享有的好处;一无所有的人,仍拥有希望。2.生活,...
戴玉手镯的好处 戴玉手镯好还是... 戴玉手镯的好处 戴玉手镯好还是碧玺好 女人戴玉?戴玉好还是碧玺好点佩戴手镯,以和田玉手镯为佳!相嫌滑...
依然什么意思? 依然什么意思?依然(汉语词语)依然,汉语词汇。拼音:yī    rán基本解释:副词,指照往常、依旧...
高尔基的散文诗 高尔基的散文诗《海燕》、《大学》、《母亲》、《童年》这些都是比较出名的一些代表作。
心在飞扬作者简介 心在飞扬作者简介心在飞扬作者简介如下。根据相关公开资料查询,心在飞扬是一位优秀的小说作者,他的小说作...
卡什坦卡的故事赏析? 卡什坦卡的故事赏析?讲了一只小狗的故事, 我也是近来才读到这篇小说. 作家对动物的拟人描写真是惟妙...
林绍涛为简艾拿绿豆糕是哪一集 林绍涛为简艾拿绿豆糕是哪一集第三十二集。 贾宽认为是阎帅间接导致刘映霞住了院,第二天上班,他按捺不...
小爱同学是女生吗小安同学什么意... 小爱同学是女生吗小安同学什么意思 小爱同学,小安同学说你是女生。小安是男的。
内分泌失调导致脸上长斑,怎么调... 内分泌失调导致脸上长斑,怎么调理内分泌失调导致脸上长斑,怎么调理先调理内分泌,去看中医吧,另外用好的...
《魔幻仙境》刺客,骑士人物属性... 《魔幻仙境》刺客,骑士人物属性加点魔幻仙境骑士2功1体质
很喜欢她,该怎么办? 很喜欢她,该怎么办?太冷静了!! 太理智了!爱情是需要冲劲的~不要考虑着考虑那~否则缘...
言情小说作家 言情小说作家我比较喜欢匪我思存的,很虐,很悲,还有梅子黄时雨,笙离,叶萱,还有安宁的《温暖的玄》 小...
两个以名人的名字命名的风景名胜... 两个以名人的名字命名的风景名胜?快太白楼,李白。尚志公园,赵尚志。
幼儿教育的代表人物及其著作 幼儿教育的代表人物及其著作卡尔威特的《卡尔威特的教育》,小卡尔威特,他儿子成了天才后写的《小卡尔威特...
海贼王中为什么说路飞打凯多靠霸... 海贼王中为什么说路飞打凯多靠霸气升级?凯多是靠霸气升级吗?因为之前刚到时确实打不过人家因为路飞的实力...
运气不好拜财神有用吗运气不好拜... 运气不好拜财神有用吗运气不好拜财神有没有用1、运气不好拜财神有用。2、拜财神上香前先点蜡烛,照亮人神...