- 浏览: 534314 次
文章分类
最新评论
android Ant 批量多渠道打包 总结!
最近研究android里用ant打多个渠道的apk包,终于搞出头绪!!走了很多错路,希望大家不要这样子!!下面就是个人的几点总结,希望对大家有帮助:
首先说的是我用的ant不是eclipse和android SDk里面自带的ant,而是从这里下载的 -- Ant官网http://ant.apache.org/, 下载Ant,当然第一步就是安装ant了,步骤如下:
第一步:安装 ant ,解压并配置环境变量
1) 解压Ant,比如解压到D:\ant
2) 我的电脑->属性->高级->环境变量
3) 系统变量新建ANT_HOME,变量值为d:\ant
4) 系统变量新建或修改path,变量值为%ANT_HOME%\bin
大家一定要把ant配置好,否则下面的都是扯淡了,无法顺利完成,测试ant是否成功,你可以在控制台输入Cmd 回车, ant 回车
如果出现
Buildfile: build.xml does not exist!
Build failed
恭喜你已经ant安装完毕了!!
第二步:简单了解Android打包步骤就行了,其他的都是扯淡,咱们直奔主题,打包前的准备
利用Ant批量打包的基本思想是,每次打包后自动替换渠道号,然后再次打包
从而实现多渠道打包的目的
这样带来了一个问题:Ant不支持循环,怎样循环打包?
扩展包Ant-contrib能轻松解决这个问题
可以的同学可以到http://ant-contrib.sourceforge.net/自行下载,或者去我的blog里的资源里下载,
另外Ant-contrib的<var>标签使用也比原来的变量方便
从而达到仅使用build.xml来实现批量打包
安装方法:直接把ant-contrib-1.0b3.jar放到Ant的lib文件夹即可
第三步:编写build.xml,直接上代码!!
<project name="Test" default="release"> <!-- ANT环境变量 --> <property environment="env" /> <!-- 使用第三方的ant包,使ant支持for循环--> <taskdef resource="net/sf/antcontrib/antcontrib.properties"> <classpath> <pathelement location="${env.ANT_HOME}/lib/ant-contrib-1.0b3.jar" /> </classpath> </taskdef> <!-- 应用名称 --> <property name="appName" value="${ant.project.name}" /> <!-- SDK目录(获取操作系统环境变量ANDROID_SDK_HOME的值) --> <property name="sdk-folder" value="${env.ANDROID_SDK_HOME}" /> <!-- SDK指定平台目录 --> <property name="sdk-platform-folder" value="${sdk-folder}/platforms/android-8" /> <!-- SDK中tools目录 --> <property name="sdk-tools" value="${sdk-folder}/tools" /> <!-- SDK指定平台中tools目录 --> <property name="sdk-platform-tools" value="${sdk-folder}/platform-tools" /> <!-- 使用到的命令(当前系统为windows,如果系统为linux,可将.bat文件替换成相对应的命令) --> <property name="aapt" value="${sdk-platform-tools}/aapt" /> <property name="aidl" value="${sdk-platform-tools}/aidl" /> <property name="dx" value="${sdk-platform-tools}/dx.bat" /> <property name="apkbuilder" value="${sdk-tools}/apkbuilder.bat" /> <property name="jarsigner" value="${env.JAVA_HOME}/bin/jarsigner" /> <property name="zipalign" value="${sdk-tools}/zipalign.exe" /> <!-- 编译需要的jar; 如果项目使用到地图服务则需要maps.jar --> <property name="android-jar" value="${sdk-platform-folder}/android.jar" /> <property name="android-maps-jar" value="${sdk-folder}/add-ons/addon_google_apis_google_inc_8/libs/maps.jar" /> <!-- --> <property name="channelname" value="" /> <property name="channelkey" value="" /> <!-- 渠道名:渠道号 --> <property name="key" value="UM:aaaaaa,BAI:bbbb" /> <!-- 编译aidl文件所需的预处理框架文件framework.aidl --> <property name="framework-aidl" value="${sdk-platform-folder}/framework.aidl" /> <!-- 清单文件 --> <property name="manifest-xml" value="AndroidManifest.xml" /> <!-- 源文件目录 --> <property name="resource-dir" value="res" /> <property name="asset-dir" value="assets" /> <!-- java源文件目录 --> <property name="srcdir" value="src" /> <property name="srcdir-ospath" value="${basedir}/${srcdir}" /> <!-- 外部类库所在目录 --> <property name="external-lib" value="libs" /> <property name="external-lib-ospath" value="${basedir}/${external-lib}" /> <!-- 版本 --> <property name="version" value="3.0" /> <!--循环打包 --> <target name="deploy"> <foreach target="modify_manifest" list="${key}" param="nameandchannel" delimiter=","> </foreach> </target> <target name="modify_manifest"> <!-- 获取渠道名字 --> <propertyregex override="true" property="channelname" input="${nameandchannel}" regexp="(.*):" select="\1" /> <!-- 获取渠道号码 --> <propertyregex override="true" property="channelkey" input="${nameandchannel}" regexp=":(.*)" select="\1" /> <!-- 正则匹配替换渠道号 --> <replaceregexp flags="g" byline="false" encoding="UTF-8"> <regexp pattern='meta-data android:value="(.*)" android:name="app_key"' /> <substitution expression='meta-data android:value="${channelkey}" android:name="app_key"' /> <fileset dir="" includes="AndroidManifest.xml" /> </replaceregexp> <antcall target="zipalign" /> </target> <!-- 初始化工作 --> <target name="init"> <echo>目录初始化....</echo> <!-- 生成R文件的相对目录 --> <var name="outdir-gen" value="gen" /> <!-- 编译后的文件放置目录 --> <var name="outdir-bin" value="bin-${channelname}" /> <!-- 生成class目录 --> <var name="outdir-classes" value="${outdir-bin}" /> <var name="outdir-classes-ospath" value="${basedir}/${outdir-classes}" /> <!-- classes.dex相关变量 --> <var name="dex-file" value="classes.dex" /> <var name="dex-path" value="${outdir-bin}/${dex-file}" /> <var name="dex-ospath" value="${basedir}/${dex-path}" /> <!-- 经过aapt生成的资源包文件 --> <var name="resources-package" value="${outdir-bin}/resources.ap_" /> <var name="resources-package-ospath" value="${basedir}/${resources-package}" /> <!-- 未认证apk包 --> <var name="out-unsigned-package" value="${outdir-bin}/${appName}-unsigned.apk" /> <var name="out-unsigned-package-ospath" value="${basedir}/${out-unsigned-package}" /> <!-- 证书文件 --> <var name="keystore-file" value="${basedir}/test.keystore" /> <!-- 已认证apk包 --> <var name="out-signed-package" value="${outdir-bin}/${appName}-${channelname}-${version}.apk" /> <var name="out-signed-package-ospath" value="${basedir}/${out-signed-package}" /> <delete dir="${outdir-bin}" /> <mkdir dir="${outdir-bin}" /> <mkdir dir="${outdir-classes}" /> </target> <!-- 根据工程中的资源文件生成R.java文件 --> <target name="gen-R" depends="init"> <echo>生成R.java文件....</echo> <exec executable="${aapt}" failonerror="true"> <arg value="package" /> <arg value="-f" /> <arg value="-m" /> <arg value="-J" /> <arg value="${outdir-gen}" /> <arg value="-S" /> <arg value="${resource-dir}" /> <arg value="-M" /> <arg value="${manifest-xml}" /> <arg value="-I" /> <arg value="${android-jar}" /> </exec> </target> <!-- 编译aidl文件 --> <target name="aidl" depends="gen-R"> <echo>编译aidl文件....</echo> <apply executable="${aidl}" failonerror="true"> <!-- 指定预处理文件 --> <arg value="-p${framework-aidl}" /> <!-- aidl声明的目录 --> <arg value="-I${srcdir}" /> <!-- 目标文件目录 --> <arg value="-o${outdir-gen}" /> <!-- 指定哪些文件需要编译 --> <fileset dir="${srcdir}"> <include name="**/*.aidl" /> </fileset> </apply> </target> <!-- 将工程中的java源文件编译成class文件 --> <target name="compile" depends="aidl"> <echo>java源文件编译成class文件....</echo> <javac encoding="utf-8" target="1.5" srcdir="." destdir="${outdir-classes}" bootclasspath="${android-jar}" verbose="false"> <compilerarg line="-encoding GBK " /> <classpath> <fileset dir="${external-lib}" includes="*.jar" /> </classpath> </javac> </target> <!-- 将.class文件转化成.dex文件 --> <target name="dex" depends="compile"> <echo>将.class文件转化成.dex文件....</echo> <exec executable="${dx}" failonerror="true"> <arg value="--dex" /> <!-- 输出文件 --> <arg value="--output=${dex-ospath}" /> <!-- 要生成.dex文件的源classes和libraries --> <arg value="${outdir-classes-ospath}" /> <arg value="${external-lib-ospath}" /> </exec> </target> <!-- 将资源文件放进输出目录 --> <target name="package-res-and-assets"> <echo>将资源文件放进输出目录....</echo> <exec executable="${aapt}" failonerror="true"> <arg value="package" /> <arg value="-f" /> <arg value="-M" /> <arg value="${manifest-xml}" /> <arg value="-S" /> <arg value="${resource-dir}" /> <arg value="-A" /> <arg value="${asset-dir}" /> <arg value="-I" /> <arg value="${android-jar}" /> <arg value="-F" /> <arg value="${resources-package}" /> </exec> </target> <!-- 打包成未签证的apk --> <target name="package" depends="dex, package-res-and-assets"> <echo>打包成未签证的apk....</echo> <exec executable="${apkbuilder}" failonerror="true"> <arg value="${out-unsigned-package-ospath}" /> <arg value="-u" /> <arg value="-z" /> <arg value="${resources-package-ospath}" /> <arg value="-f" /> <arg value="${dex-ospath}" /> <arg value="-rf" /> <arg value="${srcdir-ospath}" /> </exec> </target> <!-- 对apk进行签证 --> <target name="jarsigner" depends="package"> <echo>Packaging signed apk for release...</echo> <exec executable="${jarsigner}" failonerror="true"> <arg value="-keystore" /> <arg value="${keystore-file}" /> <arg value="-storepass" /> <arg value="123456" /> <arg value="-keypass" /> <arg value="123456" /> <arg value="-signedjar" /> <arg value="${out-signed-package-ospath}" /> <arg value="${out-unsigned-package-ospath}" /> <!-- 不要忘了证书的别名 --> <arg value="test.keystore" /> </exec> </target> <!-- 发布 --> <target name="release" depends="jarsigner"> <!-- 删除未签证apk --> <delete file="${out-unsigned-package-ospath}" /> <echo>APK is released. path:${out-signed-package-ospath}</echo> </target> <!-- 打包的应用程序进行优化 --> <target name="zipalign" depends="release"> <exec executable="${zipalign}" failonerror="true"> <arg value="-v" /> <arg value="4" /> <arg value="${out-signed-package-ospath}" /> <arg value="${out-signed-package-ospath}-zipaligned.apk" /> </exec> </target> </project>
具体操作:
首先,定义渠道名字和号码的固定格式为 渠道名字:渠道号 这样可以利用:做一个正则匹配分别获取渠道名字和渠道号,好处是最后能根据渠道名字来修改打出来的jar包名字 然后利用正则替换来替换渠道号为上面获取的渠道号,再执行一次打包动作。
根据以上的build.xml 读者只需要进行以下修改即可:
1.第一行的 <project name="Test" default="release"> name换成你自己的工程名字
2.SDK目录(获取操作系统环境变量ANDROID_SDK_HOME的值),这里的ANDROID_SDK_HOME是你的android SDK的环境变量 因为env.ANT_HOME 调用的就是你配置好的系统的环境变量,我的 ANDROID_SDK_HOME 为:E:\android\android-sdk-windows-1.5_r11\android-sdk-windows-1.5_r11 记住这里的目录一直到你的 android SDK 里很多子目录为止!因为,举个例子说吧 ,你也看到了在build.xml中用到了 ${sdk-folder}/platforms/android-8 这个就是指 在你定义的ANDROID_SDK_HOME对应的目录下去找 platforms 文件夹,所以这个目录层次很重要!!
3.<property name="jarsigner" value="${env.JAVA_HOME}/bin/jarsigner" /> 这句话 相信大家都知道了吧,因为 android 编译需要 java编译器支持,所以 JAVA_HOME 是定义的java的环境变量 ,我的电脑右键--属性 ---高级里的环境变量---新建环境变量---变量名:JAVA_HOME ,变量值:E:\Program Files\Java\jdk1.6.0_02
4.<!-- 渠道名:渠道号 --> <property name="key" value="UM:aaaaaa,BAI:bbbb" /> 我例子是:友盟和百度的两个渠道的
5.<!-- 版本 --> <property name="version" value="3.0" /> 版本号
6.重点来了 循环打包:它是怎么循环的呢? 这句话:<target name="deploy">
<foreach target="modify_manifest" list="${key}" param="nameandchannel" delimiter=",">
</foreach>
</target>
就是循环了,将${key}对应的值(也就是前面定义的<property name="key" value="UM:aaaaaa,BAI:bbbb" />) 按照delimiter="," 来分隔,进行分别打包,在modify_manifest里,这样做的:
<target name="modify_manifest">
<!-- 获取渠道名字 -->
<propertyregex override="true" property="channelname" input="${nameandchannel}" regexp="(.*):" select="\1" />
<!-- 获取渠道号码 -->
<propertyregex override="true" property="channelkey" input="${nameandchannel}" regexp=":(.*)" select="\1" />
<!-- 正则匹配替换渠道号 -->
<replaceregexp flags="g" byline="false" encoding="UTF-8">
<regexp pattern='meta-data android:value="(.*)" android:name="app_key"' />
<substitution expression='meta-data android:value="${channelkey}" android:name="app_key"' />
<fileset dir="" includes="AndroidManifest.xml" />
</replaceregexp>
<antcall target="zipalign" />
</target>
注意:我们还得需要在 AndroidManifest.xml 中默认顶一个meta节点,因为咱们再ant中是不断去替换AndroidManifest.xml中的渠道号来,达到打很多包的,所以我们还必须这样在AndroidManifest.xml中 <meta-data android:value="bbbb" android:name="app_key"></meta-data> 写一个默认的渠道,注意 app_key 这里是渠道号,一般无关紧要不需要改,只要改value="bbbb"值就可以了。
7.到这里 差不多了就,但是呢很多人打包呢,感觉不对啊,呵呵因为很多时候咱们直接运行build.xml是第一个默认运行的target不是咱们需要的,你可以仔细看看这个 build.xml的依赖关系,你就能知道了,咱们这个build.xml需要第一个运行的是 deploy的这个target,所以 需要这么做: build.xml右键,run as , 这时候不是选择运行 而是 最后那个选项: external tools configurations ,然后进去一个界面
然后接着如下图操作:
最后一下:run 搞定!!
最后还要说的就是:不同的渠道生成不同的目录,在同一渠道你会发现有两个apk ,例如:一个是Test-BAI-3.0.apk, Test-BAI-3.0.apk-zipaligned.apk ,我个人认为一个是 没有优化的,另一个是优化过的!!用优化过的吧 也就是Test-BAI-3.0.apk-zipaligned.apk 吧!
就写到这里吧!!希望对大家有些帮助!!谢谢!!
不明白的随时留言!!!祝好!
PS:对于apk的混淆,这里附加加介绍!--->点这里跳转
相关推荐
本资源将详细介绍如何使用Ant进行批量多渠道打包Android应用。 首先,你需要在你的项目中集成Ant。这通常意味着在项目根目录下创建一个名为`build.xml`的Ant构建文件。这个文件是Ant的配置中心,它定义了构建过程中...
多渠道打包通常是指针对不同平台、设备或应用市场生成多个定制版本的过程。例如,在移动应用开发中,可能需要为不同的Android设备或iOS设备创建特定的APK或IPA文件。Ant提供了一种灵活的方式,通过属性变量和条件...
而通过集成构建工具Ant,我们可以实现自动化、批量化地进行多渠道打包,大大提高工作效率。以下将详细讲解如何利用Apache Ant实现这一功能。 首先,我们需要了解Apache Ant。Ant是Java的一个构建工具,类似于Make或...
"android ant 多渠道批量打包Demo"就是一个解决此类问题的实践案例,它展示了如何通过Apache Ant工具进行自动化构建,以快速生成多个定制化的APK。 Apache Ant是一个Java库和命令行工具,它的任务是驱动构建过程。...
这就是所谓的“多渠道打包”。本文将深入探讨如何利用Apache Ant工具来实现这一目标。 Apache Ant是一个Java库和命令行工具,其主要任务是驱动构建过程。它基于XML配置文件,定义了一系列的构建任务,使得开发者...
Ant作为经典的构建工具,在Android SDK早期阶段被广泛用于自动化构建过程,包括多渠道打包。本文将详细介绍如何利用最新版的Android SDK和Ant实现这一功能。 首先,你需要确保已经安装了Android SDK,并在系统环境...
总结来说,通过ANT进行Android应用的批量打包,可以极大地提高开发效率,特别是在多渠道分发时。了解并熟练掌握ANT的使用,对于Android开发者来说是一项重要的技能。通过上述步骤和注意事项,你可以轻松地管理自己的...
总的来说,"Ant批量打包工具"是IT开发者高效、精准地进行多渠道打包的利器。它结合了Ant的强大功能和灵活性,使得构建过程可配置、可扩展,大大提升了软件项目的发布效率。通过掌握和熟练使用Ant,开发者能够更好地...
在Android多渠道打包中,主要涉及以下步骤: 1. **反编译基础APK**:使用反编译工具(如Apktool)解压原始的APK文件,将其转换回源代码的形式。这包括资源文件、Java字节码等,便于后续修改。 2. **替换渠道文件**...
总之,Ant批量打包Android应用是通过编写自定义的build.xml脚本,结合SDK工具和Ant命令,实现自动化编译和打包的过程。这可以极大地提高开发效率,特别是在需要对多版本或多渠道的应用进行打包时。
7. **多渠道打包**:在脚本中添加循环或者条件判断,遍历所有渠道的配置文件,为每个渠道生成对应的APK。 通过这样的方式,Ant批量打包脚本成为了开发团队的得力助手,让繁琐的手动打包工作变得轻松简单,也降低了...
通过学习和实践这个教程,开发者不仅可以掌握Ant批量打包的基本技巧,还能了解如何解决过程中遇到的问题,从而提高开发效率,尤其在面对多渠道发布时,能够节省大量手动操作的时间。对于大型项目而言,这种自动化...
7. **多渠道打包**:可以创建一个shell脚本或批处理文件,循环遍历各个渠道的配置,调用Ant命令进行打包,从而实现批量打包。 8. **验证和测试**:打包完成后,需要对生成的APK或IPA进行验证,确保每个渠道的打包都...
360的多渠道打包工具,正如标题所提及,是360公司提供的一款专门服务于Android应用程序开发者的工具,它允许开发者快速、方便地为不同的应用市场或者分发渠道生成特定的APK文件。这样的工具对于大型应用发行商尤其...
该配置成功对android实现了多渠道批量打包,采用的是友盟的渠道号,其他的则需要修改,还有对library项目不能直接打包,可以采用将library资源拷贝到项目中的方式曲线救国。详见:...
总结来说,Ant批量打包Android应用涉及到Android数字证书的生成、签名过程以及Ant构建脚本的配置。开发者需要掌握Java的数字证书机制,理解Ant工具的使用,以及如何根据不同的发布需求调整打包脚本。通过熟练掌握...
总的来说,Android自动化批量打渠道包是通过配置构建系统(如Gradle或Ant)来实现的,它允许开发者定义多渠道,然后通过脚本或命令行一次性完成所有渠道的打包,极大地提高了工作效率,减少了手动操作的错误。...
当你要将多个应用发布到多个推广渠道的时候,你会针对每一个渠道为android应用添加相应的渠道号,这样工作量会很大。这时你会想:如果有某种工具能针对不同的渠道批量生成apk就好了。这个时候就要求你去写个打包工具...