Ant 脚本 多渠道打包

2024-05-13 21:18
文章标签 ant 打包 脚本 多渠道

本文主要是介绍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,直接上代码!!

[java]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. <project name="Test" default="release">  
  2.     <!-- ANT环境变量 -->  
  3.     <property environment="env" />  
  4.     <!-- 使用第三方的ant包,使ant支持for循环-->  
  5.     <taskdef resource="net/sf/antcontrib/antcontrib.properties">  
  6.         <classpath>  
  7.             <pathelement location="${env.ANT_HOME}/lib/ant-contrib-1.0b3.jar" />  
  8.         </classpath>  
  9.     </taskdef>  
  10.     <!-- 应用名称 -->  
  11.     <property name="appName" value="${ant.project.name}" />  
  12.     <!-- SDK目录(获取操作系统环境变量ANDROID_SDK_HOME的值) -->  
  13.     <property name="sdk-folder" value="${env.ANDROID_SDK_HOME}" />  
  14.     <!-- SDK指定平台目录 -->  
  15.     <property name="sdk-platform-folder" value="${sdk-folder}/platforms/android-8" />  
  16.     <!-- SDK中tools目录 -->  
  17.     <property name="sdk-tools" value="${sdk-folder}/tools" />  
  18.     <!-- SDK指定平台中tools目录 -->  
  19.     <property name="sdk-platform-tools" value="${sdk-folder}/platform-tools" />  
  20.   
  21.     <!-- 使用到的命令(当前系统为windows,如果系统为linux,可将.bat文件替换成相对应的命令) -->  
  22.     <property name="aapt" value="${sdk-platform-tools}/aapt" />  
  23.     <property name="aidl" value="${sdk-platform-tools}/aidl" />  
  24.     <property name="dx" value="${sdk-platform-tools}/dx.bat" />  
  25.     <property name="apkbuilder" value="${sdk-tools}/apkbuilder.bat" />  
  26.     <property name="jarsigner" value="${env.JAVA_HOME}/bin/jarsigner" />  
  27.     <property name="zipalign" value="${sdk-tools}/zipalign.exe" />  
  28.   
  29.     <!-- 编译需要的jar; 如果项目使用到地图服务则需要maps.jar -->  
  30.     <property name="android-jar" value="${sdk-platform-folder}/android.jar" />  
  31.     <property name="android-maps-jar" value="${sdk-folder}/add-ons/addon_google_apis_google_inc_8/libs/maps.jar" />  
  32.   
  33.     <!-- -->  
  34.     <property name="channelname" value="" />  
  35.     <property name="channelkey" value="" />  
  36.     <!-- 渠道名:渠道号 -->  
  37.     <property name="key" value="UM:aaaaaa,BAI:bbbb" />  
  38.       
  39.     <!-- 编译aidl文件所需的预处理框架文件framework.aidl -->  
  40.     <property name="framework-aidl" value="${sdk-platform-folder}/framework.aidl" />  
  41.     <!-- 清单文件 -->  
  42.     <property name="manifest-xml" value="AndroidManifest.xml" />  
  43.     <!-- 源文件目录 -->  
  44.     <property name="resource-dir" value="res" />  
  45.     <property name="asset-dir" value="assets" />  
  46.     <!-- java源文件目录 -->  
  47.     <property name="srcdir" value="src" />  
  48.     <property name="srcdir-ospath" value="${basedir}/${srcdir}" />  
  49.     <!-- 外部类库所在目录 -->  
  50.     <property name="external-lib" value="libs" />  
  51.     <property name="external-lib-ospath" value="${basedir}/${external-lib}" />  
  52.   
  53.     <!-- 版本 -->  
  54.     <property name="version" value="3.0" />  
  55.       
  56.     <!--循环打包 -->  
  57.         <target name="deploy">  
  58.             <foreach target="modify_manifest" list="${key}" param="nameandchannel" delimiter=",">  
  59.             </foreach>  
  60.         </target>  
  61.         <target name="modify_manifest">  
  62.             <!-- 获取渠道名字 -->  
  63.             <propertyregex override="true" property="channelname" input="${nameandchannel}" regexp="(.*):" select="\1" />  
  64.             <!-- 获取渠道号码 -->  
  65.             <propertyregex override="true" property="channelkey" input="${nameandchannel}" regexp=":(.*)" select="\1" />  
  66.             <!-- 正则匹配替换渠道号 -->  
  67.             <replaceregexp flags="g" byline="false" encoding="UTF-8">  
  68.                 <regexp pattern='meta-data android:value="(.*)" android:name="app_key"' />  
  69.                 <substitution expression='meta-data android:value="${channelkey}" android:name="app_key"' />  
  70.                 <fileset dir="" includes="AndroidManifest.xml" />  
  71.             </replaceregexp>  
  72.             <antcall target="zipalign" />  
  73.         </target>  
  74.       
  75.     <!-- 初始化工作 -->  
  76.     <target name="init">  
  77.         <echo>目录初始化....</echo>  
  78.         <!-- 生成R文件的相对目录 -->  
  79.         <var name="outdir-gen" value="gen" />  
  80.         <!-- 编译后的文件放置目录 -->  
  81.         <var name="outdir-bin" value="bin-${channelname}" />  
  82.         <!-- 生成class目录 -->  
  83.         <var name="outdir-classes" value="${outdir-bin}" />  
  84.         <var name="outdir-classes-ospath" value="${basedir}/${outdir-classes}" />  
  85.   
  86.         <!-- classes.dex相关变量 -->  
  87.         <var name="dex-file" value="classes.dex" />  
  88.         <var name="dex-path" value="${outdir-bin}/${dex-file}" />  
  89.         <var name="dex-ospath" value="${basedir}/${dex-path}" />  
  90.   
  91.         <!-- 经过aapt生成的资源包文件 -->  
  92.         <var name="resources-package" value="${outdir-bin}/resources.ap_" />  
  93.         <var name="resources-package-ospath" value="${basedir}/${resources-package}" />  
  94.   
  95.         <!-- 未认证apk包 -->  
  96.         <var name="out-unsigned-package" value="${outdir-bin}/${appName}-unsigned.apk" />  
  97.         <var name="out-unsigned-package-ospath" value="${basedir}/${out-unsigned-package}" />  
  98.   
  99.         <!-- 证书文件 -->  
  100.         <var name="keystore-file" value="${basedir}/test.keystore" />  
  101.   
  102.         <!-- 已认证apk包 -->  
  103.         <var name="out-signed-package" value="${outdir-bin}/${appName}-${channelname}-${version}.apk" />  
  104.         <var name="out-signed-package-ospath" value="${basedir}/${out-signed-package}" />  
  105.         <delete dir="${outdir-bin}" />  
  106.         <mkdir dir="${outdir-bin}" />  
  107.         <mkdir dir="${outdir-classes}" />  
  108.     </target>  
  109.   
  110.     <!-- 根据工程中的资源文件生成R.java文件  -->  
  111.     <target name="gen-R" depends="init">  
  112.         <echo>生成R.java文件....</echo>  
  113.         <exec executable="${aapt}" failοnerrοr="true">  
  114.             <arg value="package" />  
  115.             <arg value="-f" />  
  116.             <arg value="-m" />  
  117.             <arg value="-J" />  
  118.             <arg value="${outdir-gen}" />  
  119.             <arg value="-S" />  
  120.             <arg value="${resource-dir}" />  
  121.             <arg value="-M" />  
  122.             <arg value="${manifest-xml}" />  
  123.             <arg value="-I" />  
  124.             <arg value="${android-jar}" />  
  125.         </exec>  
  126.     </target>  
  127.   
  128.     <!-- 编译aidl文件 -->  
  129.     <target name="aidl" depends="gen-R">  
  130.         <echo>编译aidl文件....</echo>  
  131.         <apply executable="${aidl}" failοnerrοr="true">  
  132.             <!-- 指定预处理文件 -->  
  133.             <arg value="-p${framework-aidl}" />  
  134.             <!-- aidl声明的目录 -->  
  135.             <arg value="-I${srcdir}" />  
  136.             <!-- 目标文件目录 -->  
  137.             <arg value="-o${outdir-gen}" />  
  138.             <!-- 指定哪些文件需要编译 -->  
  139.             <fileset dir="${srcdir}">  
  140.                 <include name="**/*.aidl" />  
  141.             </fileset>  
  142.         </apply>  
  143.     </target>  
  144.   
  145.     <!-- 将工程中的java源文件编译成class文件 -->  
  146.     <target name="compile" depends="aidl">  
  147.         <echo>java源文件编译成class文件....</echo>  
  148.         <javac encoding="utf-8" target="1.5" srcdir="." destdir="${outdir-classes}" bootclasspath="${android-jar}" verbose="false">  
  149.             <compilerarg line="-encoding GBK " />  
  150.             <classpath>  
  151.                 <fileset dir="${external-lib}" includes="*.jar" />  
  152.             </classpath>  
  153.         </javac>  
  154.     </target>  
  155.   
  156.   
  157.     <!-- 将.class文件转化成.dex文件 -->  
  158.     <target name="dex" depends="compile">  
  159.         <echo>将.class文件转化成.dex文件....</echo>  
  160.         <exec executable="${dx}" failοnerrοr="true">  
  161.             <arg value="--dex" />  
  162.             <!-- 输出文件 -->  
  163.             <arg value="--output=${dex-ospath}" />  
  164.             <!-- 要生成.dex文件的源classes和libraries -->  
  165.             <arg value="${outdir-classes-ospath}" />  
  166.             <arg value="${external-lib-ospath}" />  
  167.         </exec>  
  168.     </target>  
  169.   
  170.     <!-- 将资源文件放进输出目录 -->  
  171.     <target name="package-res-and-assets">  
  172.         <echo>将资源文件放进输出目录....</echo>  
  173.         <exec executable="${aapt}" failοnerrοr="true">  
  174.             <arg value="package" />  
  175.             <arg value="-f" />  
  176.             <arg value="-M" />  
  177.             <arg value="${manifest-xml}" />  
  178.             <arg value="-S" />  
  179.             <arg value="${resource-dir}" />  
  180.             <arg value="-A" />  
  181.             <arg value="${asset-dir}" />  
  182.             <arg value="-I" />  
  183.             <arg value="${android-jar}" />  
  184.             <arg value="-F" />  
  185.             <arg value="${resources-package}" />  
  186.         </exec>  
  187.     </target>  
  188.   
  189.     <!-- 打包成未签证的apk -->  
  190.     <target name="package" depends="dex, package-res-and-assets">  
  191.         <echo>打包成未签证的apk....</echo>  
  192.         <exec executable="${apkbuilder}" failοnerrοr="true">  
  193.             <arg value="${out-unsigned-package-ospath}" />  
  194.             <arg value="-u" />  
  195.             <arg value="-z" />  
  196.             <arg value="${resources-package-ospath}" />  
  197.             <arg value="-f" />  
  198.             <arg value="${dex-ospath}" />  
  199.             <arg value="-rf" />  
  200.             <arg value="${srcdir-ospath}" />  
  201.         </exec>  
  202.     </target>  
  203.   
  204.     <!-- 对apk进行签证 -->  
  205.     <target name="jarsigner" depends="package">  
  206.         <echo>Packaging signed apk for release...</echo>  
  207.         <exec executable="${jarsigner}" failοnerrοr="true">  
  208.             <arg value="-keystore" />  
  209.             <arg value="${keystore-file}" />  
  210.             <arg value="-storepass" />  
  211.             <arg value="123456" />  
  212.             <arg value="-keypass" />  
  213.             <arg value="123456" />  
  214.             <arg value="-signedjar" />  
  215.             <arg value="${out-signed-package-ospath}" />  
  216.             <arg value="${out-unsigned-package-ospath}" />  
  217.             <!-- 不要忘了证书的别名 -->  
  218.             <arg value="test.keystore" />  
  219.         </exec>  
  220.     </target>  
  221.   
  222.     <!-- 发布 -->  
  223.     <target name="release" depends="jarsigner">  
  224.         <!-- 删除未签证apk -->  
  225.         <delete file="${out-unsigned-package-ospath}" />  
  226.         <echo>APK is released. path:${out-signed-package-ospath}</echo>  
  227.     </target>  
  228.     <!-- 打包的应用程序进行优化 -->  
  229.       
  230.     <target name="zipalign" depends="release">  
  231.         <exec executable="${zipalign}" failοnerrοr="true">  
  232.             <arg value="-v" />  
  233.             <arg value="4" />  
  234.             <arg value="${out-signed-package-ospath}" />  
  235.             <arg value="${out-signed-package-ospath}-zipaligned.apk" />  
  236.         </exec>  
  237.     </target>  
  238. </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 脚本 多渠道打包的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/986849

相关文章

Python pyinstaller实现图形化打包工具

《Pythonpyinstaller实现图形化打包工具》:本文主要介绍一个使用PythonPYQT5制作的关于pyinstaller打包工具,代替传统的cmd黑窗口模式打包页面,实现更快捷方便的... 目录1.简介2.运行效果3.相关源码1.简介一个使用python PYQT5制作的关于pyinstall

javafx 如何将项目打包为 Windows 的可执行文件exe

《javafx如何将项目打包为Windows的可执行文件exe》文章介绍了三种将JavaFX项目打包为.exe文件的方法:方法1使用jpackage(适用于JDK14及以上版本),方法2使用La... 目录方法 1:使用 jpackage(适用于 JDK 14 及更高版本)方法 2:使用 Launch4j(

Linux使用nohup命令在后台运行脚本

《Linux使用nohup命令在后台运行脚本》在Linux或类Unix系统中,后台运行脚本是一项非常实用的技能,尤其适用于需要长时间运行的任务或服务,本文我们来看看如何使用nohup命令在后台... 目录nohup 命令简介基本用法输出重定向& 符号的作用后台进程的特点注意事项实际应用场景长时间运行的任务服

如何使用 Bash 脚本中的time命令来统计命令执行时间(中英双语)

《如何使用Bash脚本中的time命令来统计命令执行时间(中英双语)》本文介绍了如何在Bash脚本中使用`time`命令来测量命令执行时间,包括`real`、`user`和`sys`三个时间指标,... 使用 Bash 脚本中的 time 命令来统计命令执行时间在日常的开发和运维过程中,性能监控和优化是不

bat脚本启动git bash窗口,并执行命令方式

《bat脚本启动gitbash窗口,并执行命令方式》本文介绍了如何在Windows服务器上使用cmd启动jar包时出现乱码的问题,并提供了解决方法——使用GitBash窗口启动并设置编码,通过编写s... 目录一、简介二、使用说明2.1 start.BAT脚本2.2 参数说明2.3 效果总结一、简介某些情

springboot3打包成war包,用tomcat8启动

1、在pom中,将打包类型改为war <packaging>war</packaging> 2、pom中排除SpringBoot内置的Tomcat容器并添加Tomcat依赖,用于编译和测试,         *依赖时一定设置 scope 为 provided (相当于 tomcat 依赖只在本地运行和测试的时候有效,         打包的时候会排除这个依赖)<scope>provided

Linux服务器Java启动脚本

Linux服务器Java启动脚本 1、初版2、优化版本3、常用脚本仓库 本文章介绍了如何在Linux服务器上执行Java并启动jar包, 通常我们会使用nohup直接启动,但是还是需要手动停止然后再次启动, 那如何更优雅的在服务器上启动jar包呢,让我们一起探讨一下吧。 1、初版 第一个版本是常用的做法,直接使用nohup后台启动jar包, 并将日志输出到当前文件夹n

centos6一键安装vsftpd脚本

centos6一键安装vsftpd脚本 手动安装vsftpd参考教程:Centos下安装Vsftpd的图文教程 vsftpd脚本功能: 1.安装 (命令执行:sh xxx.sh)2.添加ftp用户 (命令执行:sh xxx.sh add)3.卸载vsftpd (命令执行:sh xxx.sh uninstall) 测试环境:centos6 x64 centos6 x86(测试centos7以

Android逆向(反调,脱壳,过ssl证书脚本)

文章目录 总结 基础Android基础工具 定位关键代码页面activity定位数据包参数定位堆栈追踪 编写反调脱壳好用的脚本过ssl证书校验抓包反调的脚本打印堆栈bilibili反调的脚本 总结 暑假做了两个月的Android逆向,记录一下自己学到的东西。对于app渗透有了一些思路。 这两个月主要做的是代码分析,对于分析完后的持久化等没有学习。主要是如何反编译源码,如何找到

Lua 脚本在 Redis 中执行时的原子性以及与redis的事务的区别

在 Redis 中,Lua 脚本具有原子性是因为 Redis 保证在执行脚本时,脚本中的所有操作都会被当作一个不可分割的整体。具体来说,Redis 使用单线程的执行模型来处理命令,因此当 Lua 脚本在 Redis 中执行时,不会有其他命令打断脚本的执行过程。脚本中的所有操作都将连续执行,直到脚本执行完成后,Redis 才会继续处理其他客户端的请求。 Lua 脚本在 Redis 中原子性的原因