如何将zip文件读取为字节数组,然后将字节数组转换回zip文件?

5

我曾试过使用以下代码将ZIP文件转换为字节数组:

private static byte[] readZipFile(String zipFnm)
// read in fnm, returning it as a single string
{

  FileInputStream fileInputStream=null;

  File file = new File(zipFnm);

  byte[] bFile = new byte[(int) file.length()];

  try{
    //convert file into array of bytes
    fileInputStream = new FileInputStream(zipFnm);
    fileInputStream.read(bFile);
    fileInputStream.close();
  }catch(Exception e){
    e.printStackTrace();
  }
  return bFile;
}  

这段代码用于将字节数组转换回ZIP格式,通过调用writeByteToZip(fnm + ".zip");

private static String writeByteToZip(String outFnm)
{
  try {
    FileOutputStream fileOuputStream = new FileOutputStream(outFnm); 
    fileOuputStream.write(bFile);
    fileOuputStream.close();            

  } catch ( IOException iox ){
    iox.printStackTrace();
  }
  return outFnm;
}  // end of writeByteToZip()

我该如何改正错误?我使用以下代码可以正确获取 zip 文件的字节长度
byte[] bzip = readZipFile(zipFnm);

int totalLen1 = bzip.length;
System.out.println("Total byte length of zip: " + totalLen1);

我只得到了一个大小为零的zip文件,并在Netbeans中运行时出现错误:

Exception in thread "AWT-EventQueue-0" java.lang.NullPointerException
    at java.io.FileOutputStream.write(FileOutputStream.java:305)
    at steg.Steg.writeByteToZip(Steg.java:402)
    at steg.Steg.save(Steg.java:292)
    at steg.frame1.jButton2ActionPerformed(frame1.java:349)
    at steg.frame1.access$300(frame1.java:24)
    at steg.frame1$4.actionPerformed(frame1.java:172)
    at javax.swing.AbstractButton.fireActionPerformed(AbstractButton.java:2018)
    at javax.swing.AbstractButton$Handler.actionPerformed(AbstractButton.java:2341)
    at javax.swing.DefaultButtonModel.fireActionPerformed(DefaultButtonModel.java:402)
    at javax.swing.DefaultButtonModel.setPressed(DefaultButtonModel.java:259)
    at javax.swing.plaf.basic.BasicButtonListener.mouseReleased(BasicButtonListener.java:252)
    at java.awt.Component.processMouseEvent(Component.java:6505)
    at javax.swing.JComponent.processMouseEvent(JComponent.java:3321)
    at java.awt.Component.processEvent(Component.java:6270)
    at java.awt.Container.processEvent(Container.java:2229)
    at java.awt.Component.dispatchEventImpl(Component.java:4861)
    at java.awt.Container.dispatchEventImpl(Container.java:2287)
    at java.awt.Component.dispatchEvent(Component.java:4687)
    at java.awt.LightweightDispatcher.retargetMouseEvent(Container.java:4832)
    at java.awt.LightweightDispatcher.processMouseEvent(Container.java:4492)
    at java.awt.LightweightDispatcher.dispatchEvent(Container.java:4422)
    at java.awt.Container.dispatchEventImpl(Container.java:2273)
    at java.awt.Window.dispatchEventImpl(Window.java:2713)
    at java.awt.Component.dispatchEvent(Component.java:4687)
    at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:707)
    at java.awt.EventQueue.access$000(EventQueue.java:101)
    at java.awt.EventQueue$3.run(EventQueue.java:666)
    at java.awt.EventQueue$3.run(EventQueue.java:664)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:76)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:87)
    at java.awt.EventQueue$4.run(EventQueue.java:680)
    at java.awt.EventQueue$4.run(EventQueue.java:678)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:76)
    at java.awt.EventQueue.dispatchEvent(EventQueue.java:677)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:211)
    at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:128)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:117)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:113)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:105)
    at java.awt.EventDispatchThread.run(EventDispatchThread.java:90)
4个回答

1
您提供的堆栈跟踪是由于bFile为空引起的。
原因很可能是您的readZipFile方法中的以下行:
byte[] bFile = new byte[(int) file.length()];

在这里,你正在为一个本地变量进行赋值,而不是你的类字段。尝试将该行替换为:
bFile = new byte[file.length()];

你的代码也展示了资源处理不佳。要么使用现有库方法来进行读取(例如 commons-io 或 Java 7 的 Files),要么编写以下代码:

FileInputStream fileInputStream = null;
try {
  fileInputStream = new //...
} finally {
  if (fileInputStream != null) {
    fileInputStream.close();
  }
}

如果您有Java 7,您可以使用try-with-resources:

try (FileInputStream fileInputStream = new FileInputStream(zipFnm)) {
  // Use your stream
}

不是 zipFnm 为空 - 异常是在引用 bfile 的那一行抛出的。 - drew moore

1
我没有检查你的代码。但我曾经处理过类似的需求,所以想分享我的代码。
读取zip文件并转换为字节数组:
public byte[] readXMLFile(){
        File file = new File("a.xml");

        byte[] b = new byte[(int) file.length()];
        try {
            FileInputStream fileInputStream = new FileInputStream(file);
            fileInputStream.read(b);
            fileInputStream.close();
        }catch (FileNotFoundException e) {
            System.out.println("File Not Found.");
            e.printStackTrace();
        }catch (IOException e1) {
            System.out.println("Error Reading The File.");
            e1.printStackTrace();
        }
        return b;
    }

从字节数组写入zip文件的方法:

public void writeXMLFile(byte[] fileContent){
        try {
            FileOutputStream fos = new FileOutputStream("abc.xml");

            fos.write(fileContent);
            fos.close();
        }catch(FileNotFoundException ex){
            System.out.println("FileNotFoundException : " + ex);
        }catch(IOException ioe)  {
            System.out.println("IOException : " + ioe);
        }
    }

希望这有所帮助。

始终在finally块中关闭您的流。 - Ring

1
问题很可能出在writeByteToZip()中的变量bFile上。请确保它不是null
在将来,当您发布会抛出异常的代码时,请使用注释标记抛出异常的行(// <-- NullPointerException here)。

0

正如已经提到的,该文件可能不存在。

然而,我只是想提一下在Java 7中可以使用的实用程序。Files类提供了:

static byte[] readAllBytes(Path path)
static Path write(Path path, byte[] bytes, OpenOption... options)

看起来正是你所需要的。无论文件是否为Zip格式,你都可以轻松获取字节并将它们用于写入其他文件。请注意,在Java 7中,Path表示所需文件的路径。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接