转载请说明出处http://blog.csdn.net/andywuchuanlong

记得上次在南昌中兴的一个项目中遇到过一个这种需求:一个app能够给多个渠道商去运营,渠道商推广出去能够获得对应的推广金额。

那么这种情况下就必需要使得这个app能够唯一的标志一个渠道商。

那个时候我们在这个项目中的解决方式是:让用户在app中手动填入渠道商的工号。我如今想想这种方式也是醉了,真不知道那个时候项目经理是怎么想的,居然会给出这种方案。

这次的项目中又遇到了这个问题:需求是这个app可以给多个渠道商去推广。渠道商可以获得推广金额。这次我提出的解决方式是:先把打包后的app解压,然后在assets文件夹中写入渠道商的唯一标识id。然后压缩app,压缩完成又一次签名app,之后就大工告成。用户在第一次进入app的时候。会把assets中的id读出来,提交到server,就完美的攻克了这个用户是此渠道商的推广所获得的用户。

首先第一步:把app解压。删除META-INF目录中的CERT.RSA和CERT.SF两个文件

第二步:读取解压后的assets文件夹中的id.txt文件,写入渠道商的id

[java] view
plain
copy

  1. File file = new File("d:/app/assets/id.txt");
  2. OutputStream outputStream = new FileOutputStream(file);
  3. outputStream.write(user.getId().toString().getBytes());
  4. outputStream.flush();
  5. outputStream.close();

第三步:压缩写入渠道商id后的全部app文件

[java] view
plain
copy

  1. ZipCompressor zc = new  ZipCompressor("d:/play.apk");
  2. zc.compressExe("d:/app/");

详细的压缩代码例如以下:

[java] view
plain
copy

  1. package com.xyc.signSystem.utils;
  2. import java.io.BufferedInputStream;
  3. import java.io.File;
  4. import java.io.FileInputStream;
  5. import java.io.FileOutputStream;
  6. import java.util.zip.CRC32;
  7. import java.util.zip.CheckedOutputStream;
  8. import org.apache.tools.zip.ZipEntry;
  9. import org.apache.tools.zip.ZipOutputStream;
  10. /**
  11. * @ClassName: ZipCompressor
  12. * @author :andywuchuanlong   QQ:312037487
  13. * @Description: 压缩文件的通用工具类-採用org.apache.tools.zip.ZipOutputStream实现。较复杂。
  14. *
  15. */
  16. public class ZipCompressor {
  17. static final int BUFFER = 8192;
  18. private File zipFile;
  19. /**
  20. * 压缩文件构造函数
  21. *
  22. * @param pathName
  23. *            压缩的文件存放文件夹
  24. */
  25. public ZipCompressor(String pathName) {
  26. zipFile = new File(pathName);
  27. }
  28. /**
  29. * 运行压缩操作
  30. *
  31. * @param srcPathName
  32. *            被压缩的文件/目录
  33. */
  34. public void compressExe(String srcPathName) {
  35. File file = new File(srcPathName);
  36. if (!file.exists()) {
  37. throw new RuntimeException(srcPathName + "不存在!");
  38. }
  39. try {
  40. FileOutputStream fileOutputStream = new FileOutputStream(zipFile);
  41. CheckedOutputStream cos = new CheckedOutputStream(fileOutputStream,
  42. new CRC32());
  43. ZipOutputStream out = new ZipOutputStream(cos);
  44. String basedir = "";
  45. compressByType(file, out, basedir);
  46. out.close();
  47. } catch (Exception e) {
  48. e.printStackTrace();
  49. throw new RuntimeException(e);
  50. }
  51. }
  52. /**
  53. * 推断是文件夹还是文件。依据类型(文件/文件夹)运行不同的压缩方法
  54. *
  55. * @param file
  56. * @param out
  57. * @param basedir
  58. */
  59. private void compressByType(File file, ZipOutputStream out, String basedir) {
  60. if (basedir.equals("play/")) {
  61. basedir = "";
  62. }
  63. /* 推断是文件夹还是文件 */
  64. if (file.isDirectory()) {
  65. this.compressDirectory(file, out, basedir);
  66. } else {
  67. this.compressFile(file, out, basedir);
  68. }
  69. }
  70. boolean isFirst = true;
  71. /**
  72. * 压缩一个文件夹
  73. *
  74. * @param dir
  75. * @param out
  76. * @param basedir
  77. */
  78. private void compressDirectory(File dir, ZipOutputStream out, String basedir) {
  79. if (!dir.exists()) {
  80. return;
  81. }
  82. if (basedir.equals("play/")) {
  83. basedir = "";
  84. }
  85. File[] files = dir.listFiles();
  86. for (int i = 0; i < files.length; i++) {
  87. /* 递归 */
  88. compressByType(files[i], out, basedir + dir.getName() + "/");
  89. }
  90. }
  91. /**
  92. * 压缩一个文件
  93. *
  94. * @param file
  95. * @param out
  96. * @param basedir
  97. */
  98. private void compressFile(File file, ZipOutputStream out, String basedir) {
  99. if (!file.exists()) {
  100. isFirst = false;
  101. return;
  102. }
  103. if (basedir.equals("play/")) {
  104. basedir = "";
  105. }
  106. try {
  107. BufferedInputStream bis = new BufferedInputStream(
  108. new FileInputStream(file));
  109. ZipEntry entry = new ZipEntry(basedir + file.getName());
  110. out.putNextEntry(entry);
  111. int count;
  112. byte data[] = new byte[BUFFER];
  113. while ((count = bis.read(data, 0, BUFFER)) != -1) {
  114. out.write(data, 0, count);
  115. }
  116. bis.close();
  117. } catch (Exception e) {
  118. throw new RuntimeException(e);
  119. }
  120. }
  121. }

第四步:压缩完成之后。此时的包是没有签名过的,所以还须要签名。签名能够使用jarsigner工具,首先我们要寻找到java的安装文件夹

[java] view
plain
copy

  1. <span style="white-space:pre">    </span>public  String getJavaPath() {
  2. String javaPath = (String) System.getenv("Path");
  3. String paths[]= javaPath.split(";");
  4. String myPath = null;
  5. for(String path:paths){
  6. if (path.contains("Java")&&!path.contains("jre")
  7. &&path.contains("bin") ){
  8. myPath = path;
  9. break;
  10. }
  11. }
  12. return myPath+"\\";
  13. }

签名:

[java] view
plain
copy

  1. <span style="white-space:pre">    </span>String javaPath = getJavaPath();
  2. Runtime rt = Runtime.getRuntime();
  3. String cmd = javaPath
  4. + "jarsigner -verbose"
  5. + " -keystore "+ keystorePath
  6. + " -storepass player"// password
  7. + " -signedjar "+signedApkPath // 签名后的apk存放位置
  8. + " -digestalg SHA1  -sigalg  MD5withRSA "
  9. + unsignedApkPath//未签名的apk
  10. + " player";// 别名
  11. Process child = rt.exec(cmd);

OK,签名成功。

更改已经签名的app中的内容的更多相关文章

  1. 解决“iOS 7 app自动更新,无法在app中向用户展示更新内容”问题

    转自cocoachina iOS 7能在后台自动app,这对开发者来说和用户都很方便,但是还是有一些缺点.用户不会知道app本次更新的内容,除非他们上到app的App Store页面去查看.开发者也会 ...

  2. 嵌入式表单字段中的内容可能被server更改以删除不安全的内容。是否要又一次载入您的页面以查看保存结果?

    嵌入式表单字段中的内容可能被server更改以删除不安全的内容.是否要又一次载入您的页面以查看保存结果?         近期有朋友问到,当他在SharePoint首页上进行编辑时.插入一段代码. 完 ...

  3. android APP 中微信分享功能实现 的总结

    //花了很长时间最终完成了微信分享功能,中间走了很多弯路,在此做一下小结,希望对在应用中使用到微信分享的朋友有所帮助. 主要问题就是下面两个: 1.为什么运行了项目之后,微信分享只是闪了一下就没有了? ...

  4. [Phonegap+Sencha Touch] 移动开发77 Cordova Hot Code Push插件实现自己主动更新App的Web内容

    原文地址:http://blog.csdn.net/lovelyelfpop/article/details/50848524 插件地址:https://github.com/nordnet/cord ...

  5. 全面分析:APP中的消息功能设计

    一.定义 APP的“消息”模块,是通过APP或手机这个客户端,围绕某个产品的功能进行交流.沟通的重要方式.这种沟通,一方是运营人员或商家,也可以是产品或系统本身,为方便说明笔者这里姑且统一简称为B端, ...

  6. ViewPager封装工具类: 轻松实现APP导航或APP中的广告栏

    相信做app应用开发的,绝对都接触过ViewPager,毕竟ViewPager的应用可以说无处不在:APP第一次启动时的新手导航页,APP中结合Fragment实现页面滑动,APP中常见的广告栏的自动 ...

  7. 替换excel模板中的内容并使用JavaMail发送邮件

    由于在公司工作,常年出差,每天都要以日报的形式向公司汇报当天的工作内容.而日报的内容大体上就只有当天工作的主要内容时变化的,其余的都是不变 的. 而我的电脑刚打开excel有点卡,因此决定使用Java ...

  8. 分享 Ionic 开发 Hybrid App 中遇到的问题以及后期发布 iOS/Android 的方方面面

    此篇文章主要整理了最近在使用 Ionic 开发 Hybrid App 过程中遇到的一些疑难点以及后期发布生成 iOS 和 Android 版本过程中的种种问题. 文章目录 Ionic 简介和项目需求介 ...

  9. 【iOS开发】在 App 中加入 AdMob 广告 - 入门介绍与编程技巧

    前言 虽然在App中加入广告来盈利是比较低级的商业化方式,但对于个人开发者或者小团队开发者来说,做出一个简单易用的免费小工具(举个栗子),在里面加入一些广告,如果用的人多,也是可以带来一些可观的收入的 ...

随机推荐

  1. jmeter非常好的博客收藏

    http://blog.sina.com.cn/s/blog_56c9b55c010148os.html

  2. 触屏版轻量级分页插件jqPagination分享

    说到HTML5和jquery上的分页问题,优秀的分页插件网上一抓一大把,然而同时适合兼容在Ipad和手机端的网站分页却不是特别多. 或许有人会说,触屏现在流行下拉底部后加载下一页内容,类似微博和QQ空 ...

  3. TOJ1840: Jack Straws 判断两线段相交+并查集

    1840: Jack Straws  Time Limit(Common/Java):1000MS/10000MS     Memory Limit:65536KByteTotal Submit: 1 ...

  4. classpath路径

    在项目被集成开发环境编译后,src目录下的东西会编译到WEB-INF/classes目录下,而WEB-INF/classes目录就是所谓的classpath. 将数据库连接配置dataSource.x ...

  5. Welcome-to-Swift-16自动引用计数(Automatic Reference Counting)

    Swift使用自动引用计数(ARC)来跟踪并管理应用使用的内存.大部分情况下,这意味着在Swift语言中,内存管理"仍然工作",不需要自己去考虑内存管理的事情.当实例不再被使用时, ...

  6. 备忘 CSS字体中英文名称对照表

    转载自:http://www.jb51.net/css/67658.html 在CSS文件中,我们常看到有些字体名称变成了乱码,这是由于编写者将中文字体的名字直接写成了中文,并且再上传或者拷贝复制的时 ...

  7. 飞行员配对方案问题(匈牙利算法+sort)

    洛谷传送门 匈牙利算法+sort 没什么好说的. ——代码 #include <cstdio> #include <cstring> #include <algorith ...

  8. 如何查找Windows上安装的DB2的端口号Port

    1.db2com打开控制台 2.db2 get dbm cfg 可以通过db2 get dbm cfg,查询数据库管理器配置参数,就可以查到端口号或端口名. 示例: $ db2 get dbm cfg ...

  9. Github管理 第二步:Eclipse+Github,管理Java Project版本(First Commit)

    1.提醒 如果你的Eclipse和本文一样操作,却出现了不同的结果和莫名其妙的错误,换个Eclipse也许更快. 我用了2个Eclipse,第一个一步一个坑,第2个非常顺利…… 所以,继Windows ...

  10. chromedriver对应的支持的Chrome版本(更新至Chrome64)

    很多网友在配置chromedriver的时候会遇到很多麻烦,在网上找了很多资料觉得这个表格不错,就给大家分享出来,希望对大家配置chrome的时候有帮助: chromedriver版本 支持的Chro ...