安卓开发完毕。对于一个开放应用而言,我们须要公布到不同的应用市场,同一时候我们也须要统计不同市场的用户下载量。

(通过启动应用后获取不同市场apk中的不同值来区分)

以下用一个详细的实例来说明:

1、在AndroidManifest.xml的application内加入meta-data标签

<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"> <meta-data
android:name="APP_CHANNEL"
android:value="${APP_CHANNEL_VALUE}" /> <activity
android:name=".MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>

2、改动build.gradle文件,在android {} 中加入

android {

    // 打包渠道List
productFlavors {
wandoujia {
manifestPlaceholders = [APP_CHANNEL_VALUE: "豌豆荚"]
}
cn360 {
manifestPlaceholders = [APP_CHANNEL_VALUE: "360"]
}
baidu {
manifestPlaceholders = [APP_CHANNEL_VALUE: "百度"]
}
tencent {
manifestPlaceholders = [APP_CHANNEL_VALUE: "应用宝"]
}
sougou {
manifestPlaceholders = [APP_CHANNEL_VALUE: "搜狗市场"]
}
} }

或者使用以下方法,直接使用flavor的name做为${APP_CHANNEL_VALUE}

android {

    // 打包渠道List
productFlavors {
wandoujia {}
cn360 {}
baidu {}
tencent {}
sougou {}
} // 批量处理,直接使用flavor的name作为APP_CHANNEL_VALUE的值
productFlavors.all { flavor ->
flavor.manifestPlaceholders = [APP_CHANNEL_VALUE: name]
} }

这样就完毕了。运行 gradle assembleRelease 喝口茶坐等便可。

完毕后,到build/outputs/apk中就能够看到各种渠道包。

有关assemble可用的命令还有:

gradle assembleDebug    //全部Debug版本号
gradle assembleRelease //全部Release版本号
gradle assembleBaidu //指定渠道的Debug和Release版本号
gradle assembleBaiduDebug //指定渠道的Debug版本号
gradle assembleBaiduRelease //指定渠道的Release版本号
gradle build //全部渠道的Debug和Release版本号

最后附上我的 MainActivity 和 build.gradle

package com.example.myandroid;

import android.app.Activity;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.os.Bundle;
import android.widget.Toast; /**
* 应用入口
*
* @author SHANHY(365384722@QQ.COM)
* @date 2015年12月30日
*/
public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
String channel = getAppMetaData("APP_CHANNEL");
if(channel != null)
Toast.makeText(this, channel, Toast.LENGTH_SHORT).show();
} /**
* 获取Application以下的metaData
*
* @param name
* @return
* @author SHANHY
* @date 2015年12月30日
*/
public String getAppMetaData(String meta_name){
try {
ApplicationInfo appInfo = this.getPackageManager()
.getApplicationInfo(getPackageName(),PackageManager.GET_META_DATA);
return appInfo.metaData.getString(meta_name);
} catch (NameNotFoundException e) {
e.printStackTrace();
}
return null;
}
}

完整的 build.gradle 脚本

buildscript {
repositories {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:2.0.0-alpha3'
}
} apply plugin: 'com.android.application' dependencies {
compile fileTree(dir: 'libs', include: '*.jar')
} android {
compileSdkVersion 14
buildToolsVersion "23.0.2" defaultConfig {
// 能够手动改动例如以下一些配置。无需改动不论什么代码便能够生成相应配置的apk
applicationId "com.example.myandroid.aaa"
minSdkVersion 8
targetSdkVersion 21
versionCode 200
versionName "2.0.0" // dex突破65535限制
multiDexEnabled true
// 默认打包渠道(官方)
manifestPlaceholders = [APP_CHANNEL_VALUE: "官方"]
} // 打包渠道List
productFlavors {
myself {
manifestPlaceholders = [APP_CHANNEL_VALUE: "官方"]
}
wandoujia {
manifestPlaceholders = [APP_CHANNEL_VALUE: "豌豆荚"]
}
cn360 {
manifestPlaceholders = [APP_CHANNEL_VALUE: "360"]
}
baidu {
manifestPlaceholders = [APP_CHANNEL_VALUE: "百度"]
}
tencent {
manifestPlaceholders = [APP_CHANNEL_VALUE: "应用宝"]
}
sougou {
manifestPlaceholders = [APP_CHANNEL_VALUE: "搜狗市场"]
}
} // 打包渠道List
//productFlavors {
// myself {}
// wandoujia {}
// cn360 {}
// baidu {}
// tencent {}
// sougou {}
//} // 批量处理,直接使用flavor的name作为APP_CHANNEL_VALUE的值(也能够不使用该方法,在productFlavors中逐一配置)
//productFlavors.all { flavor ->
// flavor.manifestPlaceholders = [APP_CHANNEL_VALUE: name]
//} lintOptions {
abortOnError false
} sourceSets {
main {
manifest.srcFile 'AndroidManifest.xml'
java.srcDirs = ['src']
resources.srcDirs = ['src']
aidl.srcDirs = ['src']
renderscript.srcDirs = ['src']
res.srcDirs = ['res']
assets.srcDirs = ['assets']
} // Move the tests to tests/java, tests/res, etc...
instrumentTest.setRoot('tests') // Move the build types to build-types/<type>
// For instance, build-types/debug/java, build-types/debug/AndroidManifest.xml, ...
// This moves them out of them default location under src/<type>/... which would
// conflict with src/ being used by the main source set.
// Adding new build types or product flavors should be accompanied
// by a similar customization.
debug.setRoot('build-types/debug')
release.setRoot('build-types/release')
} //签名信息
signingConfigs {
debug{
// No Debug Config
} release {
storeFile file("xxxxx.key")
storePassword "xxxxx"
keyAlias "xxxxx"
keyPassword "xxxxx"
}
}
buildTypes {
//Debug模式
debug {
// 显示LOG,在java代码中的调用方式为:BuildConfig.LOG_DEBUG。AS工具能够在BuildConfig.java中新增这个字段,假设还要兼容使用eclipse,不建议使用新增字段,由于eclipse在clean后会又一次生成BuildConfig.java(默认使用BuildConfig.DEBUG能满足须要就不要特殊处理了)
//buildConfigField "boolean", "LOG_DEBUG", "true" versionNameSuffix "-debug"
// 不开启混淆
minifyEnabled false
// 不须要ZIP优化
zipAlignEnabled false
// 不须要资源压缩
shrinkResources false
// signingConfig
signingConfig signingConfigs.debug
} //Release模式
release {
// 不显示LOG
//buildConfigField "boolean", "LOG_DEBUG", "true" minifyEnabled true
zipAlignEnabled true
// 资源压缩,移除没用的资源文件
shrinkResources true
// 混淆文件配置
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
// 签名信息配置(假设上面配置了defaultConfig则能够不用指定signingConfig)
signingConfig signingConfigs.release applicationVariants.all { variant ->
variant.outputs.each { output ->
def outputFile = output.outputFile
if (outputFile != null && outputFile.name.endsWith('.apk')) {
// 输出apk名称为myandroid_v1.0.0_2015-12-30_baidu.apk
def fileName = "myandroid_v${defaultConfig.versionName}_${releaseTime()}_${variant.productFlavors[0].name}.apk"
output.outputFile = new File(outputFile.parent, fileName)
}
}
}
}
}
} // 声明一个方法,获取打包时间
def releaseTime() {
return new Date().format("yyyy-MM-dd", TimeZone.getTimeZone("UTC"))
}

Android 使用 Gradle 多渠道打包的更多相关文章

  1. Android Studio + gradle多渠道打包

    通过工具栏的Build->Build Apk 好像只能打包第一个Module(eclipse里面是Project的概念),怎么多渠道打包呢?目前好像只能一个一个的打 首先在清单文件里设置个变量: ...

  2. [转]Android Studio系列教程六--Gradle多渠道打包

    转自:http://www.stormzhang.com/devtools/2015/01/15/android-studio-tutorial6/ Android Studio系列教程六--Grad ...

  3. Android Studio系列教程六--Gradle多渠道打包

    Android Studio系列教程六--Gradle多渠道打包 2015 年 01 月 15 日 DevTools 本文为个人原创,欢迎转载,但请务必在明显位置注明出处!http://stormzh ...

  4. Android Studio下多渠道打包

    Android Studio下实现多渠道打包 直接上步骤 步骤 1. 清单文件添加属性(以友盟统计为例) 在application标签下添加meta-data属性 <application -- ...

  5. gradle多渠道打包及友盟统计-eclipse版本

    在进行渠道打包的之前,先看看准备篇,知道打包为了什么. 参见1.0android批量打包认知. eclipse中使用gradle多渠道打包全部流程 一.生成gradle文件,配置本地的gradle环境 ...

  6. 【转】Android Studio系列教程六--Gradle多渠道打包

    原文链接:http://stormzhang.com/devtools/2015/01/15/android-studio-tutorial6/ 由于国内Android市场众多渠道,为了统计每个渠道的 ...

  7. 【Android Studio探索之路系列】之十:Gradle项目构建系统(四):Android Studio项目多渠道打包

    作者:郭孝星 微博:郭孝星的新浪微博 邮箱:allenwells@163.com 博客:http://blog.csdn.net/allenwells github:https://github.co ...

  8. Android Studio 使用Gradle多渠道打包

    第一步:配置AndroidManifest.xml 以友盟渠道为例,渠道信息一般都是写在 AndroidManifest.xml文件中,大约如下: <meta-data android:name ...

  9. android studio学习----gradle多渠道打包

    由于国内Android市场众多渠道,为了统计每个渠道的下载及其它数据统计,就需要我们针对每个渠道单独打包,如果让你打几十个市场的包岂不烦死了,不过有了Gradle,这再也不是事了. 友盟多渠道打包 废 ...

随机推荐

  1. Codeforces 1099 D. Sum in the tree-构造最小点权和有根树 贪心+DFS(Codeforces Round #530 (Div. 2))

    D. Sum in the tree time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  2. hihoCoder #1871 : Heshen's Account Book-字符串暴力模拟 自闭(getline()函数) (ACM-ICPC Asia Beijing Regional Contest 2018 Reproduction B) 2018 ICPC 北京区域赛现场赛B

    P2 : Heshen's Account Book Time Limit:1000ms Case Time Limit:1000ms Memory Limit:512MB Description H ...

  3. HDU 6280 From Tree to Graph(2018 湘潭邀请 E题,树的返祖边)

    其实打返祖边就相当于$x$到祖先这一段点(不包括两端)答案都要减$1$. 然后每个点最多减$1$次$1$. #include <bits/stdc++.h> using namespace ...

  4. Java 创建线程的方法

    为了偷懒少敲几个字这里我写了一个Util类: package test; public class Util { static void println() {System.out.println() ...

  5. Flask实战第57天:UEditor编辑器集成以及配置上传文件到七牛

    相关链接 UEditor:http://ueditor.baidu.com/website/​ 下载地址:http://ueditor.baidu.com/website/download.html# ...

  6. bytes2HexString

    public static String bytes2HexString(byte[] b) { String r = ""; for (int i = 0; i < b.l ...

  7. [Luogu2540][NOIP2016]斗地主增强版(搜索+DP)

    增强版就是原版中两鬼不算对子的版本. 先爆搜出完所有对子,剩下的牌DP处理. 考虑每个数码的拆牌情况,最多可能被拆成5种情况:1+1+1+1,1+1+2,1+3,2+2,4.故DP状态数最多为5^13 ...

  8. BZOJ 1132 [POI2008]Tro(极角排序)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1132 [题目大意] 平面上有N个点. 求出所有以这N个点为顶点的三角形的面积和(N&l ...

  9. bzoj 3140: [Hnoi2013]消毒

    3140: [Hnoi2013]消毒 Description 最近在生物实验室工作的小T遇到了大麻烦. 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为a*b*c,a.b.c 均为正整数 ...

  10. MYSQL复习笔记11-排序分组

    Date: 20140223Auth: Jin 一.排序 order by作用:对查询结果进行排序1.基本语法SELECT column_name(s) FROM table_name ORDER B ...