web-dev-qa-db-fra.com

Création d'archive Zip en Java

J'ai un fichier créé par le programme 7Zip. J'ai utilisé la méthode deflate pour le compresser. Maintenant, je veux créer la même archive (avec le même MD5sum) dans Java. Lorsque je crée un fichier Zip, j'ai utilisé l'algorithme que j'ai trouvé sur Internet, par exemple http://www.kodejava.org/examples/119.html mais lorsque j'ai créé un fichier Zip avec cette méthode, la taille compressée supérieure à la taille du fichier non compressé, que se passe-t-il? Ce n'est pas une compression très utile. Alors, comment puis-je créer un fichier Zip exactement identique au fichier Zip que j'ai créé avec le programme 7Zip? Si cela aide, j'ai toutes les informations sur le fichier Zip que j'ai créé dans le programme 7Zip.

32
hudi
// simplified code for Zip creation in Java

import Java.io.*;
import Java.util.Zip.*;

public class ZipCreateExample {

    public static void main(String[] args) throws Exception {

        // input file 
        FileInputStream in = new FileInputStream("F:/sometxt.txt");

        // out put file 
        ZipOutputStream out = new ZipOutputStream(new FileOutputStream("F:/tmp.Zip"));

        // name the file inside the Zip  file 
        out.putNextEntry(new ZipEntry("zippedjava.txt")); 

        // buffer size
        byte[] b = new byte[1024];
        int count;

        while ((count = in.read(b)) > 0) {
            out.write(b, 0, count);
        }
        out.close();
        in.close();
    }
}
59
Nishanth Thomas

Juste pour clarifier, vous avez utilisé l'algorithme Zip dans 7Zip pour votre original? De plus, 7Zip prétend avoir un taux de compression supérieur de 2 à 10% à celui des autres fournisseurs. Je pense que l’algorithme Zip intégré à Java n’est pas aussi optimisé que celui de 7Zip. Le mieux est d’appeler 7Zip à partir de la ligne de commande si vous voulez un fichier compressé de la même manière.

Essayez-vous de décompresser un fichier Zip, de modifier un fichier qu'il contient, puis de le compresser à nouveau pour qu'il ait le même hachage MD5? Les hachages sont destinés à vous empêcher de le faire. 

6
Andrew T Finnell

ZipOutputStream a peu de méthodes pour ajuster la compression:

public void setMethod (méthode int)

Définit la méthode de compression par défaut pour les entrées suivantes. Ce défaut sera utilisé chaque fois que la compression méthode n'est pas spécifiée pour un entrée individuelle du fichier Zip, et est initialement réglé sur DEFLATED.

public void setLevel (niveau int)

Définit le niveau de compression pour les entrées suivantes qui sont gonflées . Le paramètre par défaut est DEFAULT_COMPRESSION. level - le niveau de compression (0-9) 

Lorsque vous ajoutez après quelque chose comme:

ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(target));
zos.setMethod( ZipOutputStream.DEFLATED );
zos.setLevel( 5 );
...

cela n'améliore-t-il pas votre compression?

6
Maxym

Voici une fonction que vous passez le chemin absolu il va créer un fichier Zip avec le même nom que le répertoire (sous lequel vous voulez Zip de tous les sous-dossiers et fichiers, tout !!) et renvoyer true en cas de succès et false en exception si seulement.

public class FileUtil { 
final static int BUFFER = 2048;
private static Logger log = Logger.getLogger(FileUtil.class);

      public static boolean createZipArchive(String srcFolder) {

    try {
        BufferedInputStream Origin = null;



        FileOutputStream    dest = new FileOutputStream(new File(srcFolder+ ".Zip"));

        ZipOutputStream out = new ZipOutputStream(new BufferedOutputStream(dest));
        byte data[] = new byte[BUFFER];

        File subDir = new File(srcFolder);
        String subdirList[] = subDir.list();
        for(String sd:subdirList)
        {
                // get a list of files from current directory
                File f = new File(srcFolder+"/"+sd);
                if(f.isDirectory())
                {
                    String files[] = f.list();

                    for (int i = 0; i < files.length; i++) {
                        System.out.println("Adding: " + files[i]);
                        FileInputStream fi = new FileInputStream(srcFolder  + "/"+sd+"/" + files[i]);
                        Origin = new BufferedInputStream(fi, BUFFER);
                        ZipEntry entry = new ZipEntry(sd +"/"+files[i]);
                        out.putNextEntry(entry);
                        int count;
                        while ((count = Origin.read(data, 0, BUFFER)) != -1) {
                            out.write(data, 0, count);
                            out.flush();
                        }

                    }
                }
                else //it is just a file
                {
                    FileInputStream fi = new FileInputStream(f);
                    Origin = new BufferedInputStream(fi, BUFFER);
                    ZipEntry entry = new ZipEntry(sd);
                    out.putNextEntry(entry);
                    int count;
                    while ((count = Origin.read(data, 0, BUFFER)) != -1) {
                        out.write(data, 0, count);
                        out.flush();
                    }

                }
        }
        Origin.close();
        out.flush();
        out.close();
    } catch (Exception e) {
        log.info("createZipArchive threw exception: " + e.getMessage());        
        return false;

    }


    return true;
}   
  }
4
Steer360

Pour générer deux fichiers Zip identiques (y compris md5sum identique) à partir du même fichier source, je vous recommande d'utiliser le même utilitaire Zip - utilisez toujours le même programme Java ou utilisez toujours 7Zip.

Par exemple, l'utilitaire 7Zip a beaucoup d'options - dont beaucoup sont simplement des valeurs par défaut pouvant être personnalisées (ou différentes entre les versions?) - et toute implémentation Java Zip devrait également définir explicitement ces options. Si votre application Java peut simplement appeler un programme "7z" externe, vous obtiendrez de toute façon de meilleures performances qu'une implémentation Java Zip personnalisée. (Il s'agit également d'un bon exemple de problème de réduction de carte où vous pouvez facilement faire évoluer la mise en œuvre.)

Mais le problème principal que vous rencontrerez si vous avez un fichier Zip généré côté serveur et un fichier Zip généré côté client est que le fichier Zip stocke deux choses en plus du fichier original: (1) le nom du fichier, et (2) l'horodatage du fichier. Si l'un ou l'autre a changé, le fichier Zip résultant aura un md5 différent différent:

$ ls tst1/
foo.tar

$ cp -r tst1 tst2

$ ( cd tst1; Zip foo.Zip foo.tar )  ; ( cd tst2; Zip foo.Zip foo.tar )   ; md5sum tst?/foo.Zip
updating: foo.tar (deflated 20%)
updating: foo.tar (deflated 20%)
359b82678a2e17c1ddbc795ceeae7b60  tst1/foo.Zip
b55c33c0414ff987597d3ef9ad8d1d08  tst2/foo.Zip

Mais, en utilisant "cp -p" (préserver l’horodatage): 

$ cp -p -r tst1 tst2

$ ( cd tst1; Zip foo.Zip foo.tar )  ; ( cd tst2; Zip foo.Zip foo.tar )   ; md5sum tst?/foo.Zip
updating: foo.tar (deflated 20%)
updating: foo.tar (deflated 20%)
359b82678a2e17c1ddbc795ceeae7b60  tst1/foo.Zip
359b82678a2e17c1ddbc795ceeae7b60  tst2/foo.Zip

Vous trouverez le même problème avec des noms de fichiers et des chemins différents, même lorsque les fichiers contenus dans le fichier Zip sont identiques.

1
michael

S'il vous plaît trouver dans le code ci-dessous ayant les fonctionnalités à Zip *: français et unzip. J'espère que cela peut aider quelqu'un.

package com.util;

import Java.io.File;
import Java.io.FileInputStream;
import Java.io.FileOutputStream;
import Java.io.IOException;
import Java.util.ArrayList;
import Java.util.Date;
import Java.util.List;
import Java.util.Zip.ZipEntry;
import Java.util.Zip.ZipInputStream;
import Java.util.Zip.ZipOutputStream;


/**
 * @author dinesh.lomte
 *
 */
public class ZipUtil {

    /**
     * 
     * @param source
     * @param destination
     */
    public static void unZip(String source, String destination) {

        String method = "unZip(String source, String destination)";
        ZipInputStream zipInputStream = null;
        try {
            // Creating the ZipInputStream instance from the source file
            zipInputStream = new ZipInputStream(new FileInputStream(source));
            // Getting the zipped file list entry
            ZipEntry zipEntry = zipInputStream.getNextEntry();
            // Iterating through the file list entry
            while (zipEntry != null) {
                String fileName = zipEntry.getName();
                File file = new File(new StringBuilder(destination)
                    .append(File.separator)
                    .append(AppUtil.getFileNameWithoutExtension(
                            AppUtil.getNameFromPath(source)))
                    .append(File.separator).append(fileName).toString());                
                // Creating non existing folders to avoid any FileNotFoundException 
                // for compressed folder
                new File(file.getParent()).mkdirs();
                FileOutputStream fileOutputStream = new FileOutputStream(file);
                byte[] buffer = new byte[1024];
                int length;
                while ((length = zipInputStream.read(buffer)) > 0) {
                    fileOutputStream.write(buffer, 0, length);
                }
                fileOutputStream.close();
                zipEntry = zipInputStream.getNextEntry();
            }
        } catch (IOException iOException) {
            System.out.println("Failed to unzip the ''{0}'' file located in ''{1}'' folder. Due to, {2}");

        } finally {
            // Validating if zipInputStream instance in not null
            if (zipInputStream != null) {
                try {
                    zipInputStream.closeEntry();
                    zipInputStream.close();
                } catch (IOException iOException) {                 
                }
            }
        }
    }

    /**
     * Traverse a directory from the source folder location and get all files,
     * and add the file into files list.
     *
     * @param node
     */
    public static void generateFileList(
            String source, File node, List<String> files) {     
        // Validating if the node is a file
        if (node.isFile()) {
            files.add(generateZipEntry(
                    source, node.getPath().toString()));
        }
        // Validating if the node is a directory
        if (node.isDirectory()) {
            String[] subNote = node.list();
            for (String filename : subNote) {
                generateFileList(source, new File(node, filename), files);
            }
        }
    }

    /**
     * Format the file path to Zip
     * @param source
     * @param file
     * @return
     */
    private static String generateZipEntry(String source, String file) {
        return file.substring(source.length(), file.length());
    }

    /**
     * 
     * @param source
     * @param destination
     */
    public static void Zip(String source, String destination) {

        String method = "Zip(String source, String destination)";
        ZipOutputStream zipOutputStream = null;        
        try {            
            // Creating the zipOutputStream instance
            zipOutputStream = new ZipOutputStream(
                    new FileOutputStream(destination));
            List<String> files = new ArrayList<>();
            generateFileList(source, new File(source), files);
            // Iterating the list of file(s) to Zip/compress
            for (String file : files) {
                // Adding the file(s) to the Zip
                ZipEntry zipEntry = new ZipEntry(file);
                zipOutputStream.putNextEntry(zipEntry);
                FileInputStream fileInputStream = new FileInputStream(
                        new StringBuilder(source).append(File.separator)
                        .append(file).toString());
                int length;
                byte[] buffer = new byte[1024];
                while ((length = fileInputStream.read(buffer)) > 0) {
                    zipOutputStream.write(buffer, 0, length);
                }                
                // Closing the fileInputStream instance
                fileInputStream.close();
                // De-allocating the memory by assigning the null value
                fileInputStream = null;
            }
        } catch (IOException iOException) {
            System.out.println("Failed to Zip the file(s) located in ''{0}'' folder. Due to, {1}");
        } finally {
            // Validating if zipOutputStream instance in not null
            if (zipOutputStream != null) {
                try {
                    zipOutputStream.closeEntry();
                    zipOutputStream.close();
                } catch (IOException iOException) {
                }
            }
        }
    }
}
0
Dinesh Lomte
package comm;

import Java.io.File;
import Java.io.FileInputStream;
import Java.io.FileOutputStream;*emphasized text*
import Java.io.IOException;
import Java.util.Zip.ZipEntry;
import Java.util.Zip.ZipOutputStream;

public class Zip1 {
      public static void main( String[] args )
        {
            byte[] buffer = new byte[1024];

            try{

                File f= new  File("E:\\");
                f.mkdirs();
                File origFile= new File(f,"MyZipFile2.Zip");
                FileOutputStream fos = new FileOutputStream(origFile);

                ZipOutputStream zos = new ZipOutputStream(fos);
                ZipEntry ze= new ZipEntry("test.pdf");
                zos.putNextEntry(ze);
                FileInputStream in = new FileInputStream("D:\\Test.pdf");

                int len;
                while ((len = in.read(buffer)) > 0) {
                    zos.write(buffer, 0, len);
                }

                in.close();
                zos.closeEntry();

                //remember close it
                zos.close();

                System.out.println("Done");

            }catch(IOException ex){
               ex.printStackTrace();
            }
        }
}
0
vamsi