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.
// 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();
}
}
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.
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?
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;
}
}
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.
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) {
}
}
}
}
}
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();
}
}
}