`

Java之apk 解压、修改、打包、签名(2)

阅读更多
简介: 
    上篇文章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
0
0
分享到:
评论
1 楼 eleven20070525 2014-12-16  
为什么一直报无法找到文件 java.io.FileNotFoundException 异常

相关推荐

    apk解压后打包签名工具

    本篇将详细介绍如何使用apktool和Auto-Sign这两个工具进行APK的解压、打包以及签名操作。 ### APKtool:反编译与重新打包 **1. 反编译APK** APKtool是一个强大的工具,用于反编译APK文件,以便查看、修改和重新...

    apk打包工具(java)

    - **重新打包**:将修改后的文件重新组合,生成新的APK,这个过程包括Dex文件的重新生成、资源的重新打包以及APK的签名。 然而,值得注意的是,虽然apktools提供了一定的便利,但它并不适合所有的打包需求。对于...

    图形化apk打包解包签名工具apktool.rar

    使用apktool,可以轻松地将APK解压到指定目录,方便进行二次开发或分析。 二、apk反编译 Apktool能够反编译APK中的Java字节码(Dalvik字节码)为Smali代码,这是一种与Java语法类似的汇编语言。通过查看Smali代码,...

    APK的解包打包并且签名用到的Jar以及命令

    为了能够对APK进行修改,比如添加自定义功能或调试,我们需要了解如何解包、修改和重新打包APK,以及最后的签名过程。下面将详细介绍这个流程涉及的关键知识点。 1. APK解包:解包APK是为了查看和编辑其内部结构。...

    apk友盟多渠道打包java实现

    2. **Java实现打包脚本**:使用Java编写脚本可以灵活控制打包过程,包括读取渠道列表、替换资源文件、签名和zipalign等步骤。通过Java,我们可以方便地整合各种工具和库,如apktool、signapk等。 3. **apktool工具*...

    apk解包打包工具

    5. **重新打包APK**:完成修改后,使用Apktool的打包功能,执行`apktool b yourapk`命令。这会生成一个新的APK文件,但此时的APK尚未签名,无法在设备上安装。 6. **生成签名文件并签名APK**:在Android平台上,每...

    apktool 编译打包apk 签名

    3. **再编译**:修改完成后,Apktool会重新打包APK,将修改后的Smali代码和资源文件编译回DEX和资源文件,生成新的APK。 **二、Apktool的使用步骤** 1. **下载与安装**:首先,你需要从官方网站或者其他可靠的源...

    apk解包打包修改汉化工具软件

    打包时,还需要对APK进行签名,因为Android系统要求每个安装的应用必须有有效的签名。 6. **安全与合法性**:尽管这些工具提供了强大的灵活性,但擅自修改他人的APK文件可能违反版权法。只有在拥有源代码授权或者...

    Java解析apk所需jar包

    - **Apktool**: 尽管不是JAR包,但Apktool是一个流行的开源工具,可以解压、反编译、重新打包APK,其依赖的JAR包如`apktool.jar`可以用于Java项目。 - **dexlib2**或**JD-GUI**: 这些库可以帮助你读取和理解.dex文件...

    APK重新签名

    2. **解压APK**:使用像7-Zip或WinRAR这样的工具,将APK文件解压为ZIP格式。 3. **删除旧签名**:在解压后的文件夹中,找到`META-INF`目录,删除其中的`CERT.RSA`、`CERT.SF`和`MANIFEST.MF`文件,这些是原有的签名...

    apk打包解包工具

    "apk打包解包工具"是一类用于处理APK文件的实用程序,它们可以帮助开发者进行APK的构建、拆解、修改和重新签名等操作。下面将详细阐述相关知识点: 1. **APK打包**:APK打包过程是将Android应用的所有资源(如Java...

    apk重签名工具

    4. **签名新APK**:使用Java的 jarsigner 工具或专门的APK签名工具,使用你的私钥对APK进行签名。例如,使用jarsigner命令行工具,输入以下命令: ``` jarsigner -verbose -keystore myKeystore -alias myAlias -...

    APK打包解包签名工具集

    在Android应用开发中,APK打包、解包和签名是至关重要的步骤,这些操作涉及到应用程序的构建、调试和发布流程。下面将详细解释这些概念及其重要性。 首先,我们要了解APK是什么。APK(Android Package)是Android...

    apk解包+修改源码+重新打包工具

    2. 重新签名:Android系统要求每个APK都需要有自己的签名,所以重新打包后的APK需要使用新的签名文件进行签名,可以使用 jarsigner 工具完成。 3. 对齐优化:使用 zipalign 工具进行对齐优化,提高应用性能。 4. ...

    apk反编译编译签名工具包.rar

    反编译是将APK中的Java字节码转换回可读的源代码(通常是Java代码),以便于理解应用的工作原理、查看代码逻辑或修改行为。这个工具包可能包含如JADX、Apktool等工具,它们能够解析APK的DEX文件,重构出类结构和方法...

    apk自动签名工具

    2. **防止篡改**:一旦APK被签名,任何修改都会导致签名失效,防止恶意篡改。 3. **权限控制**:更新已安装的应用时,系统会检查新旧APK的签名是否一致,确保更新由原开发者提供。 4. **加密通信**:签名过程可以...

    修改apk源码,apk打包工具

    总之,修改apk源码并重新打包是一个涉及反编译、源码编辑、签名和优化的过程。这个过程中需要掌握多种工具的使用,同时也需要对Android应用的结构和生命周期有清晰的理解。在实际操作中,一定要注意保留原始APK的...

    apktool反编译解包,打包签名,修改代码工具全套

    在Android应用开发领域,了解如何使用工具对APK文件进行反编译、解包、打包、修改代码以及签名是至关重要的技能。APKtool就是这样一个强大的工具,它为开发者提供了深入洞察APK内部结构和修改应用程序的能力。接下来...

Global site tag (gtag.js) - Google Analytics