android 多渠道打包
android应用上传时,需要区分开每个渠道。一般都会在配置文件中更改一个渠道id,如果有多个渠道,手动修改并生成apk的话会非常麻烦,而且增大出错概率。
在这分享一个打包工具类.
我们项目中使用的umeng做统计分析工具, umeng在分渠道打包的时候需要修改manifest.xml中的
<meta-data android:name="UMENG_CHANNEL" android:value="@string/channel_name" /> value值。
该值放到了strings.xml中<string name="channel_name">app_channel</string>,所以在编译的时候只需修改app_channel。
下面是工具类,直接运行即可.
package com.yang.main; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.io.OutputStreamWriter; /** * @description: android渠道打包工具类,确保路径填写无误,渠道名称填入channels中 * @author: yangyang@duitang.com * @time: 2012-10-23下午5:17:47 */ public class CompiledApkUpdate { private static final String androidSDK_PATH = "D:\\android\\android-sdk-windows\\"; //android SDK路径 public static final String APK_NAME = "duitang.apk"; public static final String PROJECT_LIBARY = ""; public static final String PROJECT_PATH = "D:\\android\\workspace\\tmp\\duitang\\"; //要打包的工程路径 public static final String APK_PATH = "D:\\android\\workspace\\tmp\\apk\\duitang_"; //打包后存放apk的路径 duitang_是前缀 private static final String apk_PATH_keystore = "D:\\android\\keystore\\bb"; //apk签名文件路径 private static final String channelFlag = "app_channel"; // public static String[] channels = {"duitang"}; private static String currentChannelName = ""; public static String[] channels = {"duitang","91","360","QQ","jifeng","anzhuo","anzhi","youyi","appchina","wangyi","baidu","souhu","3g","nduo","xiaomi","huawei","meizu","lianxiang","aliyun","taobao","google","nearme","mumayi","wandoujia","crosscat","dangle","maopao","feiliu"}; public static void main(String[] args) { replaceChannel(); } /** * 替换渠道名称 */ public static void replaceChannel() { try { String outPath = PROJECT_PATH + "res\\values\\strings.xml"; // 输出文件位置 String content = read(outPath); for(int channelid=0;channelid<channels.length;channelid++){ String tmpContent = content; tmpContent = tmpContent.replaceFirst(channelFlag, channels[channelid]); currentChannelName = channels[channelid]; write(tmpContent,outPath); System.out.println("replace channel name over..."); packageRes(); // 一次渠道号的更改完成。可以进行打包了。 createUnsignedApk(); signedApk(channelid); } write(content,outPath); //完成后还原channel_name System.out.println("execute over!"); } catch (Exception e) { e.printStackTrace(); } } /** * class文件打包成classes.dex */ public static void packageDex(){ try { System.out.println("dx.bat start..."); Process process = Runtime.getRuntime().exec(androidSDK_PATH +"platform-tools\\dx.bat --dex --output=" +PROJECT_PATH+"bin\\classes.dex " +PROJECT_PATH+"bin\\classes " +PROJECT_PATH+"libs\\*.jar"); new MyThread(process.getErrorStream()).start(); new MyThread(process.getInputStream()).start(); process.waitFor(); process.destroy(); System.out.println("dx.bat over..."); } catch (Exception e) { e.printStackTrace(); } } /** * res assets文件打包成res.zip */ public static void packageRes(){ try{ System.out.println(currentChannelName+" create resources.ap"); String library = ""; if(PROJECT_LIBARY!=null&&!PROJECT_LIBARY.equals("")){ library = "-S " + PROJECT_LIBARY + "res "; } Process process = null; process = Runtime .getRuntime() .exec( androidSDK_PATH + "platform-tools\\aapt.exe package -f " + "-M " + PROJECT_PATH + "AndroidManifest.xml " + //-M 指定配置文件 "-S " + PROJECT_PATH + "res " + //-S 指定资源文件 library + "-A " + PROJECT_PATH + "assets " + //-A 指定assets "-I " + androidSDK_PATH + "platforms\\android-16\\android.jar " + //-I 指定android类 "-F " + PROJECT_PATH + "bin\\resources.ap_ --auto-add-overlay"); // 将资源文件打包resources.ap_ new MyThread(process.getErrorStream()).start(); new MyThread(process.getInputStream()).start(); process.waitFor(); process.destroy(); System.out.println(currentChannelName+" resources.ap over..."); }catch(Exception e){ e.printStackTrace(); } } /** * classes.dex res.zip AndroidManifest.xml组合成未签名的apk */ public static void createUnsignedApk(){ try{ System.out.println(currentChannelName+" createUnsignedApk start"); Process process = null; process = Runtime.getRuntime().exec( androidSDK_PATH+ "tools\\apkbuilder.bat " + PROJECT_PATH + "bin\\"+APK_NAME+" -u -z " + PROJECT_PATH + "bin\\resources.ap_ -f " + PROJECT_PATH + "bin\\classes.dex"); // 生成未签名的apk new MyThread(process.getErrorStream()).start(); new MyThread(process.getErrorStream()).start(); process.waitFor(); process.destroy(); System.out.println(currentChannelName+" createUnsignedApk over"); }catch(Exception e){ e.printStackTrace(); } } /** * 签名apk */ public static void signedApk(int channelid){ try{ System.out.println(currentChannelName+" signed apk start"); Process process = null; String jarsigner; jarsigner = "jarsigner -keystore "+apk_PATH_keystore+" -storepass ***** -keypass ****** " + "-signedjar " + APK_PATH + channels[channelid] + ".apk " + PROJECT_PATH + "bin\\"+APK_NAME+" *****"; //签名apk process = Runtime .getRuntime() .exec(jarsigner); // 对apk进行签名 new MyThread(process.getErrorStream()).start(); new MyThread(process.getInputStream()).start(); process.waitFor(); process.destroy(); System.out.println(currentChannelName+" signed apk over"); // 一条渠道的打包完成。文件会输出到指定目录 }catch(Exception e){ e.printStackTrace(); } } public static String read(String path) { StringBuffer res = new StringBuffer(); String line = null; try { BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(path),"UTF-8")); while ((line = reader.readLine()) != null) { res.append(line + "\n"); } reader.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } return res.toString(); } public static boolean write(String cont, String dist) { try { OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(new File(dist)),"utf-8"); writer.write(cont); writer.flush(); writer.close(); return true; } catch (IOException e) { e.printStackTrace(); return false; } } public static class MyThread extends Thread{ BufferedReader bf; public MyThread(InputStream input) { bf = new BufferedReader(new InputStreamReader(input)); } public void run() { String line; try { line = bf.readLine(); while (line != null) { System.out.println(line); line = bf.readLine(); } } catch (IOException e) { e.printStackTrace(); } } } }
注意事项:
1. 正确设置各个path路径.
2. channelFlag的值要与 strings.xml 中channel_name的value一致.
3. 运行java文件是先运行一下android工程,确保bin目录下有最新的apk.
4. 运行过程中不要手动停止,如手动停止,再次运行时请检查strings.xml中channel_name的value值,如不为app_channel,请修改
5. 如有error: Error parsing XML: not well-formed (invalid token)类似错误,注意设置字符编码
相关推荐
本文将深入探讨Android多渠道打包工具及其工作原理。 一、多渠道打包的概念 多渠道打包是指在构建APK时,通过替换或插入特定的渠道标识符,使同一个应用可以根据不同分发渠道进行定制。这样做的好处在于,开发者...
下面我们将详细探讨如何实现Android多渠道打包,以及如何替换资源文件、logo、包名和项目名。 首先,我们来看如何进行多渠道打包。传统的做法是通过在build.gradle文件中添加多个productFlavors,每个flavor对应一...
总结起来,Android多渠道打包是通过定制化APK来追踪和分析不同推广渠道的效果。通过合理地利用Gradle或者第三方工具,可以有效地提高打包效率,减少手动操作带来的错误,同时确保每个渠道的APK都能准确地携带相应的...
一、Android多渠道打包 1. 多渠道打包的概念:多渠道打包是指同一个应用根据不同的推广渠道(如应用市场、广告联盟等)生成带有不同标识的APK,以便追踪不同渠道的安装来源和效果。这通常通过修改AndroidManifest....
本教程将围绕“Android多渠道打包”这一主题,以友盟SDK为例,讲解如何实现这一功能。 首先,理解多渠道打包的背景和目的。在Android应用的分发过程中,开发者可能需要与多个推广渠道合作,每个渠道可能有自己的...
Android多渠道打包方案通常涉及到替换APK中的特定资源或配置文件,以便针对每个渠道进行定制。在Android P(即Android 9.0)中,系统引入了一些新特性和限制,对传统的多渠道打包方式提出了挑战。本文将详细介绍如何...
本文将详细介绍Android多渠道打包工具及其工作原理,帮助开发者理解如何利用这类工具提高工作效率。 标题中的“android多渠道打包工具”指的是一个用于自动化创建多个定制化APK的程序。这种工具通常基于美团的多...
Android多渠道打包工具,如标题所提及的"Android-Android多渠道打包工具",是一个用于创建针对不同渠道定制apk的解决方案。这种工具的主要目标是允许开发者在APK中嵌入特定渠道的标识,以便追踪应用的下载来源。 在...
在Android应用开发中,多渠道...总的来说,Walle是一款高效、易用的Android多渠道打包工具,能够帮助开发者快速适应各种分发需求,提高开发效率。了解并掌握Walle的使用,对于Android开发者来说是一项非常实用的技能。
Android多渠道打包是应用发布的重要环节,ApkMultiChannelPlugin提供了一种便捷的方式,使得开发者能够轻松地为不同渠道创建定制化APK。通过理解这个插件的工作原理,开发者可以更高效地管理自己的分发策略,提升...
Android多渠道打包工具(安卓多渠道资源工具)多渠道包装工具Android多渠道打包工具(安卓多渠道资源工具)@deprecated本工具以落后,推荐使用美团多渠道资源工具APK签名方案v2,新一代开源Android渠道包生成工具...
在Android应用开发中,发布应用通常需要针对不同的市场或渠道进行定制化打包,这被称为多渠道打包。每个渠道可能需要不同的广告配置、统计代码或者其他特定的设置。"android 多渠道打包工具"就是为了方便开发者高效...
一个 Mac App,用于 Apk 多渠道打包。 使用美团的方案,在 META-INF 目录写入空文件 支持 AES 加密,CBC 和 ECB 两种模式 可以自定义前缀 下载最新版本,有问题可以在 Issues 中反馈。 License GPLv3
Walle就是美团开源的一个用于Android多渠道打包的工具,它简化了这个过程,使得开发者能够更加高效地管理和打包多个渠道的apk。 Walle插件基于Gradle构建系统,Gradle是Android Studio默认的构建工具,它提供了一种...