常常服務器須要對文件進行壓縮,網絡上流傳較多的是從磁盤文件中來壓縮成zip文件。可是經常服務器的文件存放在內存中,以byte[]形式存儲在內存中。這個時候就不能使用網絡上流傳的經常使用方法了,這裏就須要對內存文件進行壓縮。經過內存文件來壓縮成zip的方式,首先性能方面比磁盤壓縮要快不少,另外內存文件壓縮到方式不會產生臨時文件或者磁盤文件,而磁盤讀取文件方式壓縮,會產生新的zip文件。而後要說下的是,內存壓縮方式能夠支持內存文件壓縮,也能夠手動將磁盤文件讀取到內存而後在進行壓縮,這樣也不會產生新的臨時文件。java
該工具類支持加密和非加密兩種壓縮方式apache
如下貼上個人代碼數組
本身擴展的內存壓縮代碼服務器
ZipUtils.java
package com.cigna.hmc.groupinsurance.utils; import java.io.IOException; import org.apache.commons.lang.StringUtils; import net.lingala.zip4j.exception.ZipException; import net.lingala.zip4j.io.ZipOutputStream; import net.lingala.zip4j.model.ZipParameters; import net.lingala.zip4j.util.Zip4jConstants; /** * * @author josnow * @date 2017年5月24日 下午3:12:31 * @version 1.0.0 * @desc zip工具集成原CompressUtil方法,增長了對內存文件和流文件的壓縮以免產生臨時文件 */ public class ZipUtils extends CompressUtil { /** * * @desc 將內存文件寫入zip內。注意:最後必須調用closeZipOutputStream關閉輸出流,或者手動關閉 * @auth josnow * @date 2017年5月24日 下午5:23:02 * @param fileName * 文件名 * @param data * 文件數據 * @param password * 密碼 */ public static void addFileToZip(String fileName, byte[] data, String password, ZipOutputStream zipOutputStream) throws ZipException, IOException { if (StringUtils.isEmpty(fileName) || data == null || data.length == 0 || zipOutputStream == null) { throw new ZipException(new StringBuilder("參數異常,fileName=").append(fileName).append(",data=").append(data) .append(",zipOutputStream=").append(zipOutputStream).toString()); } ZipParameters zipParameters = new ZipParameters(); zipParameters.setCompressionMethod(Zip4jConstants.COMP_DEFLATE); // 壓縮方式 zipParameters.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL); // 壓縮級別 zipParameters.setFileNameInZip(fileName); if (StringUtils.isNotBlank(password)) { zipParameters.setEncryptFiles(true); zipParameters.setEncryptionMethod(Zip4jConstants.ENC_METHOD_STANDARD); zipParameters.setPassword(password.toCharArray()); } // 源文件是否爲外部流,true表示使用內存文件而非本地存儲文件 zipParameters.setSourceExternalStream(true); zipOutputStream.putNextEntry(null, zipParameters); zipOutputStream.write(data); zipOutputStream.closeEntry(); } /** * * @desc 將內存文件寫入zip內。注意:最後必須調用closeZipOutputStream關閉輸出流,或者手動關閉 * @auth josnow * @date 2017年5月24日 下午5:46:02 * @param fileName * 文件名 * @param data * 文件數據 */ public static void addFileToZip(String fileName, byte[] data, ZipOutputStream zipOutputStream) throws ZipException, IOException { addFileToZip(fileName, data, null, zipOutputStream); } /** * * @desc 將內存文件寫入zip內。注意:最後必須調用closeZipOutputStream關閉輸出流,或者手動關閉 * @auth josnow * @date 2017年5月25日 上午11:08:56 * @param zipParameters * zip參數 * @param data * 文件數據 * @param zipOutputStream * 輸出流 */ public static void addFileToZip(ZipParameters zipParameters, byte[] data, ZipOutputStream zipOutputStream) throws ZipException, IOException { if (zipParameters == null || data == null || data.length == 0 || zipOutputStream == null) { throw new ZipException(new StringBuilder("參數異常,zipParameters=").append(zipParameters).append(",data=") .append(data).append(",zipOutputStream=").append(zipOutputStream).toString()); } zipOutputStream.putNextEntry(null, zipParameters); zipOutputStream.write(data); zipOutputStream.closeEntry(); } /** * * @desc 關閉流 * @auth josnow * @date 2017年5月25日 上午11:16:01 * @param zipOutputStream * 輸出流 */ public static void closeZipOutputStream(ZipOutputStream zipOutputStream) throws IOException, ZipException { if (zipOutputStream == null) { return; } zipOutputStream.finish(); zipOutputStream.close(); } // public static void main(String[] args) throws Exception { // ByteArrayOutputStream byteArrayOutputStream = new // ByteArrayOutputStream(1024); // ZipOutputStream zipOutputStream = new // ZipOutputStream(byteArrayOutputStream); // // byte[] b = "德瑪西亞哦哦奧法額外發撒旦;聯發科就;".getBytes(); // // addFileToZip("你好大豬頭.txt", b, zipOutputStream); // addFileToZip("你就是大肥豬.txt", b, "123", zipOutputStream); // // closeZipOutputStream(zipOutputStream); // // byte[] zipData = byteArrayOutputStream.toByteArray(); // System.out.println(new String(zipData)); // // new FileOutputStream("D:\\nima.zip").write(zipData); // } }
CompressUtil.java 網絡上流傳的壓縮工具類網絡
package com.cigna.hmc.groupinsurance.utils; import java.io.File; import java.util.ArrayList; import java.util.Collections; import java.util.List; import org.apache.commons.lang.StringUtils; import net.lingala.zip4j.core.ZipFile; import net.lingala.zip4j.exception.ZipException; import net.lingala.zip4j.model.FileHeader; import net.lingala.zip4j.model.ZipParameters; import net.lingala.zip4j.util.Zip4jConstants; /** * ZIP壓縮文件操做工具類 支持密碼 依賴zip4j開源項目(http://www.lingala.net/zip4j/) 版本1.3.1 * * @author ninemax */ public class CompressUtil { /** * 使用給定密碼解壓指定的ZIP壓縮文件到指定目錄 * <p> * 若是指定目錄不存在,能夠自動建立,不合法的路徑將致使異常被拋出 * * @param zip * 指定的ZIP壓縮文件 * @param dest * 解壓目錄 * @param passwd * ZIP文件的密碼 * @return 解壓後文件數組 * @throws ZipException * 壓縮文件有損壞或者解壓縮失敗拋出 */ public static File[] unzip(String zip, String dest, String passwd) throws ZipException { File zipFile = new File(zip); return unzip(zipFile, dest, passwd); } /** * 使用給定密碼解壓指定的ZIP壓縮文件到當前目錄 * * @param zip * 指定的ZIP壓縮文件 * @param passwd * ZIP文件的密碼 * @return 解壓後文件數組 * @throws ZipException * 壓縮文件有損壞或者解壓縮失敗拋出 */ public static File[] unzip(String zip, String passwd) throws ZipException { File zipFile = new File(zip); File parentDir = zipFile.getParentFile(); return unzip(zipFile, parentDir.getAbsolutePath(), passwd); } /** * 使用給定密碼解壓指定的ZIP壓縮文件到指定目錄 * <p> * 若是指定目錄不存在,能夠自動建立,不合法的路徑將致使異常被拋出 * * @param zip * 指定的ZIP壓縮文件 * @param dest * 解壓目錄 * @param passwd * ZIP文件的密碼 * @return 解壓後文件數組 * @throws ZipException * 壓縮文件有損壞或者解壓縮失敗拋出 */ public static File[] unzip(File zipFile, String dest, String passwd) throws ZipException { ZipFile zFile = new ZipFile(zipFile); zFile.setFileNameCharset("GBK"); if (!zFile.isValidZipFile()) { throw new ZipException("壓縮文件不合法,可能被損壞."); } File destDir = new File(dest); if (destDir.isDirectory() && !destDir.exists()) { destDir.mkdir(); } if (zFile.isEncrypted()) { zFile.setPassword(passwd.toCharArray()); } zFile.extractAll(dest); List<FileHeader> headerList = zFile.getFileHeaders(); List<File> extractedFileList = new ArrayList<File>(); for (FileHeader fileHeader : headerList) { if (!fileHeader.isDirectory()) { extractedFileList.add(new File(destDir, fileHeader.getFileName())); } } File[] extractedFiles = new File[extractedFileList.size()]; extractedFileList.toArray(extractedFiles); return extractedFiles; } /** * 壓縮指定文件到當前文件夾 * * @param src * 要壓縮的指定文件 * @return 最終的壓縮文件存放的絕對路徑,若是爲null則說明壓縮失敗. */ public static String zip(String src) { return zip(src, null); } /** * 使用給定密碼壓縮指定文件或文件夾到當前目錄 * * @param src * 要壓縮的文件 * @param passwd * 壓縮使用的密碼 * @return 最終的壓縮文件存放的絕對路徑,若是爲null則說明壓縮失敗. */ public static String zip(String src, String passwd) { return zip(src, null, passwd); } /** * 使用給定密碼壓縮指定文件或文件夾到當前目錄 * * @param src * 要壓縮的文件 * @param dest * 壓縮文件存放路徑 * @param passwd * 壓縮使用的密碼 * @return 最終的壓縮文件存放的絕對路徑,若是爲null則說明壓縮失敗. */ public static String zip(String src, String dest, String passwd) { return zip(src, dest, true, passwd); } /** * 使用給定密碼壓縮指定文件或文件夾到指定位置. * <p> * dest可傳最終壓縮文件存放的絕對路徑,也能夠傳存放目錄,也能夠傳null或者"".<br /> * 若是傳null或者""則將壓縮文件存放在當前目錄,即跟源文件同目錄,壓縮文件名取源文件名,以.zip爲後綴;<br /> * 若是以路徑分隔符(File.separator)結尾,則視爲目錄,壓縮文件名取源文件名,以.zip爲後綴,不然視爲文件名. * * @param src * 要壓縮的文件或文件夾路徑 * @param dest * 壓縮文件存放路徑 * @param isCreateDir * 是否在壓縮文件裏建立目錄,僅在壓縮文件爲目錄時有效.<br /> * 若是爲false,將直接壓縮目錄下文件到壓縮文件. * @param passwd * 壓縮使用的密碼 * @return 最終的壓縮文件存放的絕對路徑,若是爲null則說明壓縮失敗. */ public static String zip(String src, String dest, boolean isCreateDir, String passwd) { File srcFile = new File(src); dest = buildDestinationZipFilePath(srcFile, dest); ZipParameters parameters = new ZipParameters(); parameters.setCompressionMethod(Zip4jConstants.COMP_DEFLATE); // 壓縮方式 parameters.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL); // 壓縮級別 if (!StringUtils.isEmpty(passwd)) { parameters.setEncryptFiles(true); parameters.setEncryptionMethod(Zip4jConstants.ENC_METHOD_STANDARD); // 加密方式 parameters.setPassword(passwd.toCharArray()); } try { ZipFile zipFile = new ZipFile(dest); if (srcFile.isDirectory()) { // 若是不建立目錄的話,將直接把給定目錄下的文件壓縮到壓縮文件,即沒有目錄結構 if (!isCreateDir) { File[] subFiles = srcFile.listFiles(); ArrayList<File> temp = new ArrayList<File>(); Collections.addAll(temp, subFiles); zipFile.addFiles(temp, parameters); return dest; } zipFile.addFolder(srcFile, parameters); } else { zipFile.addFile(srcFile, parameters); } return dest; } catch (ZipException e) { e.printStackTrace(); } return null; } /** * 構建壓縮文件存放路徑,若是不存在將會建立 傳入的多是文件名或者目錄,也可能不傳,此方法用以轉換最終壓縮文件的存放路徑 * * @param srcFile * 源文件 * @param destParam * 壓縮目標路徑 * @return 正確的壓縮文件存放路徑 */ private static String buildDestinationZipFilePath(File srcFile, String destParam) { if (StringUtils.isEmpty(destParam)) { if (srcFile.isDirectory()) { destParam = srcFile.getParent() + File.separator + srcFile.getName() + ".zip"; } else { String fileName = srcFile.getName().substring(0, srcFile.getName().lastIndexOf(".")); destParam = srcFile.getParent() + File.separator + fileName + ".zip"; } } else { createDestDirectoryIfNecessary(destParam); // 在指定路徑不存在的狀況下將其建立出來 if (destParam.endsWith(File.separator)) { String fileName = ""; if (srcFile.isDirectory()) { fileName = srcFile.getName(); } else { fileName = srcFile.getName().substring(0, srcFile.getName().lastIndexOf(".")); } destParam += fileName + ".zip"; } } return destParam; } /** * 在必要的狀況下建立壓縮文件存放目錄,好比指定的存放路徑並無被建立 * * @param destParam * 指定的存放路徑,有可能該路徑並無被建立 */ private static void createDestDirectoryIfNecessary(String destParam) { File destDir = null; if (destParam.endsWith(File.separator)) { destDir = new File(destParam); } else { destDir = new File(destParam.substring(0, destParam.lastIndexOf(File.separator))); } if (!destDir.exists()) { destDir.mkdirs(); } } }