从Zip文件中的文件读取内容


74

我正在尝试创建一个简单的Java程序,该程序从zip文件中的文件读取和提取内容。压缩文件包含3个文件(txt,pdf,docx)。我需要阅读所有这些文件的内容,并且为此使用了Apache Tika

有人可以帮我实现此功能。到目前为止,我已经尝试过了,但是没有成功

程式码片段

public class SampleZipExtract {


    public static void main(String[] args) {

        List<String> tempString = new ArrayList<String>();
        StringBuffer sbf = new StringBuffer();

        File file = new File("C:\\Users\\xxx\\Desktop\\abc.zip");
        InputStream input;
        try {

          input = new FileInputStream(file);
          ZipInputStream zip = new ZipInputStream(input);
          ZipEntry entry = zip.getNextEntry();

          BodyContentHandler textHandler = new BodyContentHandler();
          Metadata metadata = new Metadata();

          Parser parser = new AutoDetectParser();

          while (entry!= null){

                if(entry.getName().endsWith(".txt") || 
                           entry.getName().endsWith(".pdf")||
                           entry.getName().endsWith(".docx")){
              System.out.println("entry=" + entry.getName() + " " + entry.getSize());
                     parser.parse(input, textHandler, metadata, new ParseContext());
                     tempString.add(textHandler.toString());
                }
           }
           zip.close();
           input.close();

           for (String text : tempString) {
           System.out.println("Apache Tika - Converted input string : " + text);
           sbf.append(text);
           System.out.println("Final text from all the three files " + sbf.toString());
        } catch (FileNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (SAXException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (TikaException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

1
为什么不直接将zip文件传递给Apache Tika?然后,它将为zip中的每个文件调用您提供的递归解析器,因此您无需执行任何特殊操作!
Gagravarr

那就是我想知道的,但是却没有足够的教程。我对此也不太担心-javamex.com/tutorials/compression/zip_problems.shtml,不确定Tika是否解决了此问题。
S Jagdeesh

提卡(Tika)使用
公用

2
蒂卡61 Mb?61 Mb仅适用于ZIP,可使用约10个字符串完成!我的应用程序有15项活动权重小于4 Mb。我认为对于用户而言,拥有仅适用于琐碎任务的大型应用程序是不屑一顾的。
Acuna

Answers:


183

如果您想知道如何从每个文件中获取文件内容,ZipEntry那实际上很简单。这是一个示例代码:

public static void main(String[] args) throws IOException {
    ZipFile zipFile = new ZipFile("C:/test.zip");

    Enumeration<? extends ZipEntry> entries = zipFile.entries();

    while(entries.hasMoreElements()){
        ZipEntry entry = entries.nextElement();
        InputStream stream = zipFile.getInputStream(entry);
    }
}

一旦拥有InputStream,就可以根据需要读取它。


22
不要忘记关闭inputStream和ZipFile以避免资源泄漏:)。
Noremac

3
zipFile.entries(); 没有为zipFile类型定义条目函数
Vasanth Nag KV

2
有没有一种方法可以将byte []数组传递给的构造函数ZipFile (content.getBytes())?如果没有,我们该怎么做?
Simple-Solution

@ Simple-Solution我认为最简单的方法是将字节数组写入new File,并将该File实例提供给构造函数
Rodrigo Sasaki 2014年

最终解决方案+1
Ved Prakash

48

从Java 7开始,NIO Api提供了一种更好和更通用的方式来访问Zip或Jar文件的内容。实际上,它现在是一个统一的API,可让您像对待普通文件一样对待Zip文件。

为了提取此API中zip文件中包含的所有文件,您需要执行以下操作:

在Java 8中:

private void extractAll(URI fromZip, Path toDirectory) throws IOException{
    FileSystems.newFileSystem(fromZip, Collections.emptyMap())
            .getRootDirectories()
            .forEach(root -> {
                // in a full implementation, you'd have to
                // handle directories 
                Files.walk(root).forEach(path -> Files.copy(path, toDirectory));
            });
}

在Java 7中:

private void extractAll(URI fromZip, Path toDirectory) throws IOException{
    FileSystem zipFs = FileSystems.newFileSystem(fromZip, Collections.emptyMap());

    for(Path root : zipFs.getRootDirectories()) {
        Files.walkFileTree(root, new SimpleFileVisitor<Path>() {
            @Override
            public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) 
                    throws IOException {
                // You can do anything you want with the path here
                Files.copy(file, toDirectory);
                return FileVisitResult.CONTINUE;
            }

            @Override
            public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) 
                    throws IOException {
                // In a full implementation, you'd need to create each 
                // sub-directory of the destination directory before 
                // copying files into it
                return super.preVisitDirectory(dir, attrs);
            }
        });
    }
}

12
这既很棒又疯狂。
Esko

2
FileSystem手术后应关闭。
艾哈迈德·阿瑟

5
在Java 8版本中,Files.walk(root)抛出无法通过lambda传播的IOException。
barteks2x

11

由于中的条件while,循环可能永远不会中断:

while (entry != null) {
  // If entry never becomes null here, loop will never break.
}

null您可以尝试以下方法来代替检查:

ZipEntry entry = null;
while ((entry = zip.getNextEntry()) != null) {
  // Rest of your code
}

我们不能只使用while(zip.getNextEntry()!= null)吗?
沙蒂尔


2

我实现此目的的方式是创建ZipInputStream包装类,该包装类将进行处理,该类仅提供当前条目的流:

包装器类:

public class ZippedFileInputStream extends InputStream {

    private ZipInputStream is;

    public ZippedFileInputStream(ZipInputStream is){
        this.is = is;
    }

    @Override
    public int read() throws IOException {
        return is.read();
    }

    @Override
    public void close() throws IOException {
        is.closeEntry();
    }

}

使用方法:

    ZipInputStream zipInputStream = new ZipInputStream(new FileInputStream("SomeFile.zip"));

    while((entry = zipInputStream.getNextEntry())!= null) {

     ZippedFileInputStream archivedFileInputStream = new ZippedFileInputStream(zipInputStream);

     //... perform whatever logic you want here with ZippedFileInputStream 

     // note that this will only close the current entry stream and not the ZipInputStream
     archivedFileInputStream.close();

    }
    zipInputStream.close();

这种方法的优点之一是:InputStreams作为参数传递给处理它们的方法,并且这些方法趋向于在完成输入流后立即关闭。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.