简介:
上篇文章
http://showlike.iteye.com/admin/blogs/1688679中,通过Runtime.getRuntime().exec 调用命令的方式对APK进行 解压、打包、签名。此文不同之处在于应用java.util.zip对APK进行解压、打包,感觉说得有点多,直接上代码。
代码实现:
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.util.Enumeration;
import java.util.zip.CRC32;
import java.util.zip.CheckedOutputStream;
import java.util.zip.ZipOutputStream;
import org.apache.tools.zip.ZipEntry;
import org.apache.tools.zip.ZipFile;
/**
* @author showlike
* @version v1.0
* 2012-9-28 上午11:03:36
* explain : 文件解压/打包工具
*/
public class ZipUtil {
private static final int BUFFER = 1024;
private static final String BASE_DIR = "";
/**符号"/"用来作为目录标识判断符*/
private static final String PATH = "/";
/**签名目录*/
private static final String SIGN_PATH_NAME = "META-INF";
/**修改文件目录*/
private static final String UPDATE_PATH_NAME = "\\res\\raw\\channel";
/**解压源文件目录*/
private static final String SOURCE_PATH_NAME = "\\source\\";
/**打包目录*/
private static final String TARGET_PATH_NAME = "\\target\\";
/**签名目录*/
private static final String RESULT_PATH_NAME = "\\result\\";
/**JDK BIN 目录*/
private static final String JDK_BIN_PATH = "C:\\Program Files\\Java\\jdk1.6.0_26\\bin";
/**密钥 目录*/
private static final String SECRET_KEY_PATH = "F:\\document\\APK\\";
/**密钥 名称*/
private static final String SECRET_KEY_NAME = "sdk.keystore";
/**
* 解压缩zip文件
* @param fileName 要解压的文件名 包含路径 如:"c:\\test.zip"
* @param filePath 解压后存放文件的路径 如:"c:\\temp"
* @throws Exception
*/
@SuppressWarnings("rawtypes")
public static void unZip(String fileName, String filePath) throws Exception{
ZipFile zipFile = new ZipFile(fileName);
Enumeration emu = zipFile.getEntries();
while(emu.hasMoreElements()){
ZipEntry entry = (ZipEntry) emu.nextElement();
if (entry.isDirectory()){
new File(filePath+entry.getName()).mkdirs();
continue;
}
BufferedInputStream bis = new BufferedInputStream(zipFile.getInputStream(entry));
File file = new File(filePath + entry.getName());
File parent = file.getParentFile();
if(parent != null && (!parent.exists())){
parent.mkdirs();
}
FileOutputStream fos = new FileOutputStream(file);
BufferedOutputStream bos = new BufferedOutputStream(fos,BUFFER);
byte [] buf = new byte[BUFFER];
int len = 0;
while((len=bis.read(buf,0,BUFFER))!=-1){
fos.write(buf,0,len);
}
bos.flush();
bos.close();
bis.close();
}
zipFile.close();
}
/**
* 压缩文件
*
* @param srcFile
* @param destPath
* @throws Exception
*/
public static void compress(String srcFile, String destPath) throws Exception {
compress(new File(srcFile), new File(destPath));
}
/**
* 压缩
*
* @param srcFile
* 源路径
* @param destPath
* 目标路径
* @throws Exception
*/
public static void compress(File srcFile, File destFile) throws Exception {
// 对输出文件做CRC32校验
CheckedOutputStream cos = new CheckedOutputStream(new FileOutputStream(
destFile), new CRC32());
ZipOutputStream zos = new ZipOutputStream(cos);
compress(srcFile, zos, BASE_DIR);
zos.flush();
zos.close();
}
/**
* 压缩
*
* @param srcFile
* 源路径
* @param zos
* ZipOutputStream
* @param basePath
* 压缩包内相对路径
* @throws Exception
*/
private static void compress(File srcFile, ZipOutputStream zos,
String basePath) throws Exception {
if (srcFile.isDirectory()) {
compressDir(srcFile, zos, basePath);
} else {
compressFile(srcFile, zos, basePath);
}
}
/**
* 压缩目录
*
* @param dir
* @param zos
* @param basePath
* @throws Exception
*/
private static void compressDir(File dir, ZipOutputStream zos,
String basePath) throws Exception {
File[] files = dir.listFiles();
// 构建空目录
if (files.length < 1) {
ZipEntry entry = new ZipEntry(basePath + dir.getName() + PATH);
zos.putNextEntry(entry);
zos.closeEntry();
}
String dirName = "";
String path = "";
for (File file : files) {
//当父文件包名为空时,则不把包名添加至路径中(主要是解决压缩时会把父目录文件也打包进去)
if(basePath!=null && !"".equals(basePath)){
dirName=dir.getName();
}
path = basePath + dirName + PATH;
// 递归压缩
compress(file, zos, path);
}
}
/**
* 文件压缩
*
* @param file
* 待压缩文件
* @param zos
* ZipOutputStream
* @param dir
* 压缩文件中的当前路径
* @throws Exception
*/
private static void compressFile(File file, ZipOutputStream zos, String dir)
throws Exception {
/**
* 压缩包内文件名定义
*
* <pre>
* 如果有多级目录,那么这里就需要给出包含目录的文件名
* 如果用WinRAR打开压缩包,中文名将显示为乱码
* </pre>
*/
if("/".equals(dir))dir="";
else if(dir.startsWith("/"))dir=dir.substring(1,dir.length());
ZipEntry entry = new ZipEntry(dir + file.getName());
zos.putNextEntry(entry);
BufferedInputStream bis = new BufferedInputStream(new FileInputStream(file));
int count;
byte data[] = new byte[BUFFER];
while ((count = bis.read(data, 0, BUFFER)) != -1) {
zos.write(data, 0, count);
}
bis.close();
zos.closeEntry();
}
public static void main(String[] args)throws Exception{
StringBuffer buffer = new StringBuffer();
BufferedReader br =null;
OutputStreamWriter osw =null;
String srcPath = "F:\\document\\APK\\new\\iGouShop.apk";
String content= "channel_id=LD20120926";
File srcFile = new File(srcPath);
String parentPath = srcFile.getParent(); //源文件目录
String fileName = srcFile.getName(); //源文件名称
String prefixName = fileName.substring(0, fileName.lastIndexOf("."));
//解压源文件保存路径
String sourcePath = buffer.append(parentPath).append(SOURCE_PATH_NAME).
append(prefixName).append("\\").toString();
//------解压
unZip(srcPath, sourcePath);
//------删除解压后的签名文件
String signPathName = sourcePath+SIGN_PATH_NAME;
File signFile = new File(signPathName);
if(signFile.exists()){
File sonFiles[] = signFile.listFiles();
if(sonFiles!=null && sonFiles.length>0){
//循环删除签名目录下的文件
for(File f : sonFiles){
f.delete();
}
}
signFile.delete();
}
//------修改内容
buffer.setLength(0);
String path = buffer.append(parentPath).append(SOURCE_PATH_NAME)
.append(prefixName).append(UPDATE_PATH_NAME).toString();
br = new BufferedReader(new InputStreamReader(new FileInputStream(path)));
while((br.readLine())!=null)
{
osw = new OutputStreamWriter(new FileOutputStream(path));
osw.write(content,0,content.length());
osw.flush();
}
//------打包
String targetPath = parentPath+TARGET_PATH_NAME;
//判断创建文件夹
File targetFile = new File(targetPath);
if(!targetFile.exists()){
targetFile.mkdir();
}
compress(parentPath+SOURCE_PATH_NAME+prefixName,targetPath+fileName);
//------签名
File ff =new File(JDK_BIN_PATH);
String resultPath = parentPath+RESULT_PATH_NAME;
//判断创建文件夹
File resultFile = new File(resultPath);
if(!resultFile.exists()){
resultFile.mkdir();
}
//组合签名命令
buffer.setLength(0);
buffer.append("cmd.exe /c jarsigner -keystore ")
.append(SECRET_KEY_PATH).append(SECRET_KEY_NAME)
.append(" -storepass winadsdk -signedjar ")
.append(resultPath).append(fileName).append(" ") //签名保存路径应用名称
.append(targetPath).append(fileName).append(" ") //打包保存路径应用名称
.append(SECRET_KEY_NAME);
//利用命令调用JDK工具命令进行签名
Process process = Runtime.getRuntime().exec(buffer.toString(),null,ff);
if(process.waitFor()!=0)System.out.println("文件打包失败!!!");
}
}
注:此实现在应用解压上有些不能通过,但在解压我司应用时未出现该异常,原因未能解决,希望大牛们给予指点。
关于java zip推荐博文:
http://snowolf.iteye.com/blog/465433#bc2283658
分享到:
相关推荐
本篇将详细介绍如何使用apktool和Auto-Sign这两个工具进行APK的解压、打包以及签名操作。 ### APKtool:反编译与重新打包 **1. 反编译APK** APKtool是一个强大的工具,用于反编译APK文件,以便查看、修改和重新...
- **重新打包**:将修改后的文件重新组合,生成新的APK,这个过程包括Dex文件的重新生成、资源的重新打包以及APK的签名。 然而,值得注意的是,虽然apktools提供了一定的便利,但它并不适合所有的打包需求。对于...
使用apktool,可以轻松地将APK解压到指定目录,方便进行二次开发或分析。 二、apk反编译 Apktool能够反编译APK中的Java字节码(Dalvik字节码)为Smali代码,这是一种与Java语法类似的汇编语言。通过查看Smali代码,...
为了能够对APK进行修改,比如添加自定义功能或调试,我们需要了解如何解包、修改和重新打包APK,以及最后的签名过程。下面将详细介绍这个流程涉及的关键知识点。 1. APK解包:解包APK是为了查看和编辑其内部结构。...
2. **Java实现打包脚本**:使用Java编写脚本可以灵活控制打包过程,包括读取渠道列表、替换资源文件、签名和zipalign等步骤。通过Java,我们可以方便地整合各种工具和库,如apktool、signapk等。 3. **apktool工具*...
5. **重新打包APK**:完成修改后,使用Apktool的打包功能,执行`apktool b yourapk`命令。这会生成一个新的APK文件,但此时的APK尚未签名,无法在设备上安装。 6. **生成签名文件并签名APK**:在Android平台上,每...
3. **再编译**:修改完成后,Apktool会重新打包APK,将修改后的Smali代码和资源文件编译回DEX和资源文件,生成新的APK。 **二、Apktool的使用步骤** 1. **下载与安装**:首先,你需要从官方网站或者其他可靠的源...
打包时,还需要对APK进行签名,因为Android系统要求每个安装的应用必须有有效的签名。 6. **安全与合法性**:尽管这些工具提供了强大的灵活性,但擅自修改他人的APK文件可能违反版权法。只有在拥有源代码授权或者...
- **Apktool**: 尽管不是JAR包,但Apktool是一个流行的开源工具,可以解压、反编译、重新打包APK,其依赖的JAR包如`apktool.jar`可以用于Java项目。 - **dexlib2**或**JD-GUI**: 这些库可以帮助你读取和理解.dex文件...
2. **解压APK**:使用像7-Zip或WinRAR这样的工具,将APK文件解压为ZIP格式。 3. **删除旧签名**:在解压后的文件夹中,找到`META-INF`目录,删除其中的`CERT.RSA`、`CERT.SF`和`MANIFEST.MF`文件,这些是原有的签名...
"apk打包解包工具"是一类用于处理APK文件的实用程序,它们可以帮助开发者进行APK的构建、拆解、修改和重新签名等操作。下面将详细阐述相关知识点: 1. **APK打包**:APK打包过程是将Android应用的所有资源(如Java...
4. **签名新APK**:使用Java的 jarsigner 工具或专门的APK签名工具,使用你的私钥对APK进行签名。例如,使用jarsigner命令行工具,输入以下命令: ``` jarsigner -verbose -keystore myKeystore -alias myAlias -...
在Android应用开发中,APK打包、解包和签名是至关重要的步骤,这些操作涉及到应用程序的构建、调试和发布流程。下面将详细解释这些概念及其重要性。 首先,我们要了解APK是什么。APK(Android Package)是Android...
2. 重新签名:Android系统要求每个APK都需要有自己的签名,所以重新打包后的APK需要使用新的签名文件进行签名,可以使用 jarsigner 工具完成。 3. 对齐优化:使用 zipalign 工具进行对齐优化,提高应用性能。 4. ...
反编译是将APK中的Java字节码转换回可读的源代码(通常是Java代码),以便于理解应用的工作原理、查看代码逻辑或修改行为。这个工具包可能包含如JADX、Apktool等工具,它们能够解析APK的DEX文件,重构出类结构和方法...
2. **防止篡改**:一旦APK被签名,任何修改都会导致签名失效,防止恶意篡改。 3. **权限控制**:更新已安装的应用时,系统会检查新旧APK的签名是否一致,确保更新由原开发者提供。 4. **加密通信**:签名过程可以...
总之,修改apk源码并重新打包是一个涉及反编译、源码编辑、签名和优化的过程。这个过程中需要掌握多种工具的使用,同时也需要对Android应用的结构和生命周期有清晰的理解。在实际操作中,一定要注意保留原始APK的...
在Android应用开发领域,了解如何使用工具对APK文件进行反编译、解包、打包、修改代码以及签名是至关重要的技能。APKtool就是这样一个强大的工具,它为开发者提供了深入洞察APK内部结构和修改应用程序的能力。接下来...