一、前言
最近公司推广项目,经常要打30多个渠道的包,打得我有点痛苦,于是决定研究批量打包,以后自动化解决,就不用那样一个个渠道慢慢搞了。
二、Ant的安装
1.下载Ant
2.解压并配置环境变量
a) 解压Ant,比如解压到D:\ant
b) 我的电脑->属性->高级->环境变量
c) 系统变量新建ANT_HOME,变量值为d:\ant
d) 系统变量新建或修改path,变量值为%ANT_HOME%\bin
3.测试
Cmd/控制台输入ant
如果出现
Buildfile: build.xml does not exist!
Build failed
表示环境变量设置成功
三、Android打包步骤
1.用aapt命令生成R.java文件
2.用aidl命令生成相应java文件
3.用javac命令编译java源文件生成class文件
4.用dx.bat将class文件转换成classes.dex文件
5.用aapt命令生成资源包文件resources.ap_
6.用apkbuilder.bat打包资源和classes.dex文件,生成unsigned.apk
7.用jarsinger命令对apk认证,生成signed.apk
8.用zipalign命令对apk进行优化
四、打包前的准备
利用Ant批量打包的基本思想是,每次打包后自动替换渠道号,然后再次打包
从而实现多渠道打包的目的
这样带来了一个问题:Ant不支持循环,怎样循环打包?
扩展包Ant-contrib能轻松解决这个问题
可以翻墙的同学可以到http://ant-contrib.sourceforge.net/自行下载
文章底部也会放出Ant-contrib的包(我也是从上面的网站下载的,纯转)
另外Ant-contrib的<var>标签使用也比原来的变量方便
从而达到仅使用build.xml来实现批量打包
安装方法:直接把ant-contrib-1.0b3.jar放到Ant的lib文件夹即可
使用方法:添加以下代码
<taskdef resource="net/sf/antcontrib/antcontrib.properties"> <classpath> <pathelement location="lib/ant-contrib-1.0b3.jar"/> </classpath> </taskdef>
五、编写build.xml 实际上网上很多地方也给出了现成的build.xml,本文很多地方也是参考自网上,但是网上的资料大多是写好一个build.xml,然后通过其他脚本来替换,这样给人一种不干净的感觉,我希望所有工作都在build.xml完成,只需敲入ant命令就能开始批量打包,于是对线程的build.xml做了一点修改。 首先,定义渠道名字和号码的固定格式为 渠道名字:渠道号 这样可以利用:做一个正则匹配分别获取渠道名字和渠道号,好处是最后能根据渠道名字来修改打出来的jar包名字 然后利用正则替换来替换渠道号为上面获取的渠道号,再执行一次打包动作 具体代码如下:
<!-- 渠道名:渠道号 --> <property name="key" value="nameA:aaaaaa,nameB:bbbbbb"/> <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>
六.Build.xml
<project name="ant" default="release"> <!-- ANT环境变量 --> <property environment="env" /> <!-- 使用第三方的ant包,使ant支持for循环--> <taskdef resource="net/sf/antcontrib/antcontrib.properties"> <classpath> <pathelement location="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指定平台目录 --> <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="" /> <!-- 编译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="key" value="nameA:aaaaaa"/> <!-- 版本 --> <property name="version" value="3.0" /> <!-- 初始化工作 --> <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}/G3_key" /> <!-- 已认证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> <!--循环打包 --> <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> <!-- 根据工程中的资源文件生成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="xxxxxx" /> <arg value="-keypass" /> <arg value="xxxxxx" /> <arg value="-signedjar" /> <arg value="${out-signed-package-ospath}" /> <arg value="${out-unsigned-package-ospath}"/> <!-- 不要忘了证书的别名 --> <arg value="xxxxxx"/> </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>
七、参考网站
http://www.cnblogs.com/qianxudetianxia/archive/2012/07/04/2573687.html
http://www.cnblogs.com/ondream/archive/2012/06/18/ant.html
http://www.cnblogs.com/xbglbc/archive/2012/02/09/2343816.html
http://blog.csdn.net/xyz_lmn/article/details/7367783
八、题外话
这个build经过测试可以成功打出对应的包,比起手动一个个换渠道号来打方便很多。
实际上我后来在Android的SDK的tools\ant里面找到了更标准的打包实例(最喜欢标准了),因为时间关系还没进行研究,有兴趣的可以进行研究并根据这个build来实现循环打包。
实际上我仅仅是对网上的几个资料进行了一次简单的修正和合并,感谢参考网站的诸位站长