Java zip 程序生成损坏的 zip 文件

标签 java zip zipoutputstream

我想用 Java 编写一个 GUI 压缩/解压缩程序。该程序将能够压缩文件和目录的任意组合,并解压缩一个或多个压缩文件。

现在我刚刚完成了 GUI 和 zip 功能。但 zip 功能似乎无法正常工作,生成的 zip 文件以某种方式损坏。我找不到问题到底出在哪里。它似乎与 compress 函数或 zipFile 函数一起使用。

当我测试该程序时,输出如下:

192-168-1-101:工具 user0$ 解压 test1.zip

存档:test1.zip 未找到中央目录结尾签名。要么这个文件不是 一个 zip 文件,或者它构成多部分存档的一个磁盘。在里面 后一种情况可以在以下位置找到中央目录和 zipfile 注释 该存档的最后一个磁盘。

unzip: 在 test1.zip 或之一中找不到 zipfile 目录 test1.zip.zip,并且找不到 test1.zip.ZIP,期间。

非常感谢您的帮助。

代码如下:

package javazip;

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.border.TitledBorder;
import javax.swing.filechooser.FileFilter;
import java.io.*;
import java.util.zip.*;

public class JZip {
    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable () {
            public void run () {
                new JZipFrame();
            }
        });
    }
}

class JZipFrame extends JFrame {
    private JTextArea displayArea;

    public JZipFrame () {
        setTitle("JZip");
        Toolkit tk = Toolkit.getDefaultToolkit();
        Dimension screenSize = tk.getScreenSize();
        int screenHeight = screenSize.height;
        int screenWidth = screenSize.width;
        setSize(screenWidth / 4, screenHeight / 4);
        setLocation(screenWidth / 4, screenHeight / 4);
        setDefaultCloseOperation(EXIT_ON_CLOSE);

        JMenuBar menuBar = createMenuBar();
        JScrollPane displayPanel = createDisplayPanel();

        add(displayPanel);
        setJMenuBar(menuBar);

        setVisible(true);
    } 

    private JMenuBar createMenuBar () {
        JMenuBar menuBar = new JMenuBar();
        JMenu menu = new JMenu("Operations");
        JMenuItem compressItem = createCompressItem();
        JMenuItem decompressItem = createDecompressItem();
        JMenuItem quitItem = createQuitItem();

        menu.add(compressItem);
        menu.add(decompressItem);
        menu.add(quitItem);
        menuBar.add(menu);

        return menuBar;
    }

    private JScrollPane createDisplayPanel () {
        displayArea = new JTextArea(20, 40);
        displayArea.setMargin(new Insets(5, 5, 5, 5));
        displayArea.setEditable(false);
        JScrollPane scrlPanel = new JScrollPane(displayArea);

        return scrlPanel;
    }

    private JMenuItem createCompressItem () {
        JMenuItem cItem = new JMenuItem("Compression");
        cItem.addActionListener(new ActionListener () {
            public void actionPerformed (ActionEvent event) {
                File[] cChosenItems = selectFilesForCompression();
                compress(cChosenItems);
            }
        });

        return cItem;
    }

    private JMenuItem createDecompressItem () {
        JMenuItem dItem = new JMenuItem("Decompression");
        dItem.addActionListener(new ActionListener () {
            public void actionPerformed (ActionEvent event) {
                File[] dChosenItems = selectFilesForDecompression();
                decompress(dChosenItems);
            }
        });

        return dItem;
    }

    private final File[] selectFilesForCompression () {
        final JFileChooser cChooser = new JFileChooser();
        File[] selectedItems = null;

        cChooser.setMultiSelectionEnabled(true);
        cChooser.setFileSelectionMode(JFileChooser.FILES_AND_DIRECTORIES);
        int cRetVal = cChooser.showOpenDialog(JZipFrame.this);
        if (cRetVal == JFileChooser.APPROVE_OPTION) {
            selectedItems = cChooser.getSelectedFiles();
            for (File item : selectedItems)
                displayArea.append(item.getParent() + System.getProperty("file.separator") 
                                        + item.getName() + "\n");
        }
        return selectedItems;
    }

    private final File[] selectFilesForDecompression () {
        final JFileChooser dChooser = new JFileChooser();
        dChooser.setMultiSelectionEnabled(true);
        dChooser.setFileSelectionMode(JFileChooser.FILES_ONLY);
        dChooser.addChoosableFileFilter(new FileFilter() {
            public String getDescription () {
                return "ZIP Files (*.zip)";
            }
            public boolean accept (File f) {
                if (f.isDirectory())
                    return true;
                else 
                    return f.getName().toLowerCase().endsWith(".zip");
            }
        });
        int dRetVal = dChooser.showOpenDialog(JZipFrame.this);
        File[] selectedItems = null;

        if (dRetVal == JFileChooser.APPROVE_OPTION) {
            selectedItems = dChooser.getSelectedFiles();
            for (File item : selectedItems)
                displayArea.append(item.getParent() + System.getProperty("file.separator") 
                                        + item.getName() + "\n");
        }
        return selectedItems;
    }

    private JMenuItem createQuitItem () {
        JMenuItem qItem = new JMenuItem("Quit");
        qItem.addActionListener(new ActionListener () {
            @Override
            public void actionPerformed (ActionEvent event) {
                System.exit(0);
            }
        });
        return qItem;
    }

    private final void compress (File[] files) {
        FileOutputStream out = null;
        String zipName = null;

        if (files.length == 1) {
            zipName = files[0].getName();
            if (files[0].isDirectory()) {
                if (zipName.endsWith("/") || zipName.endsWith("\\"))
                    zipName = zipName.substring(0, zipName.length() - 1);
            }
            try {
                String name = zipName.substring(0, zipName.lastIndexOf("."));
                out = new FileOutputStream(name + ".zip");
            }
            catch (FileNotFoundException e) {
                JOptionPane.showMessageDialog(this, e.toString(), "Error", JOptionPane.ERROR_MESSAGE);
            }
        }
        else {
            try {
                out = new FileOutputStream(files[0].getParent() + "/compressed.zip");
            }
            catch (FileNotFoundException e) {
                JOptionPane.showMessageDialog(this, e.toString(), "Error", JOptionPane.ERROR_MESSAGE);
            }
        }
        ZipOutputStream zipOut = new ZipOutputStream(out);
        zipOut.setMethod(ZipOutputStream.DEFLATED);
        for (File f : files) {
            try {
                zipFile(f, f.getName(), zipOut);    
            }
            catch (IOException e) {
                JOptionPane.showMessageDialog(this, e.toString(), "Error", JOptionPane.ERROR_MESSAGE);
            }
            displayArea.append("Now processing: " + f.getName() + "\n");
        }
        JOptionPane.showMessageDialog(this, "Compression was successful!", "Message", JOptionPane.INFORMATION_MESSAGE);
    }
    // The problem may be with this function or the one above
    private void zipFile (File file, String fileName, ZipOutputStream zipOut) throws IOException {

        if (file.isHidden()) {
            return;
        }
        if (file.isDirectory()) {
            if (fileName.endsWith("/") || fileName.endsWith("\\")) {
                zipOut.putNextEntry(new ZipEntry(fileName));
                zipOut.closeEntry();
            }
            else {
                zipOut.putNextEntry(new ZipEntry(fileName + "/"));
                zipOut.closeEntry();
            }
            File[] children = file.listFiles();
            for (File childFile : children)
                zipFile(childFile, fileName + "/" + childFile.getName(), zipOut);

            return;
        }
        FileInputStream input = new FileInputStream(file);
        ZipEntry zipEntry = new ZipEntry(fileName);
        zipOut.putNextEntry(zipEntry);
        byte[] bytes = new byte[1024];
        int length;
        while ((length = input.read(bytes)) >= 0) {
            zipOut.write(bytes, 0, length);
        }
        input.close();
    } 

    private final void decompress (File[] files) {
        // TODO
    }
}

最佳答案

在网上摸索之后,我发现了一个关于ZipOutStream的教程,其中我注意到ZipOutputStream对象在压缩操作后被关闭。因此,我添加了代码来关闭 ZipOutputStream 对象,即代码中 compress 方法中的 zipOut 。现在,代码可以正常压缩单个文件、多个文件、目录以及文件和目录的组合。

因此,在使用输入流和输出流对象时,一定要记得在使用后关闭它们。否则缓冲区将无法正确读取或写入以产生所需的结果。

我现在将继续编写decompress方法,也许还会添加更多功能。可能还会遇到问题,到时候我会来这里寻求帮助。这里的人非常乐于助人,解决了我的很多问题。非常感谢您的帮助。

再次感谢您的参与。

关于Java zip 程序生成损坏的 zip 文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61625497/

相关文章:

java - 用于序列化 Base64 编码的 Java 对象的 LONGTEXT 或 BLOB

java - UnsatisfiedLinkError:库已加载,但我仍然收到链接错误

JavaScript 创建 zip 文件

java - ZipOutputStream 离开 size=-1

java - 如何创建包含多个图像文件的 zip 文件

java - 带有消息监听器的事务处理 session ,未使用消息

java - Rest API 日志记录实践问题

dependencies - gradle zip依赖的一部分,命名fileTree

java - 如何在不写入输出流的情况下从 ZipInputStream 获取每个 ZipFile 条目的字节/内容?

java - Mac 操作系统 Chrome 浏览器 : Archive Utility (Error 1) - Operation not permitted: Unable to expand downloaded zip