java zip4j 內存文件和磁盤文件 壓縮和加密

常常服務器須要對文件進行壓縮,網絡上流傳較多的是從磁盤文件中來壓縮成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();
        }
    }

}
相關文章
相關標籤/搜索