我有一个zip文件,其中包含我想要阅读的9个文件。所以我实现了以下功能:
public static Map<String, BufferedReader> unzipFile(InputStream zippedFile) throws IOException {
ZipInputStream zipInputStream = new ZipInputStream(zippedFile);
HashMap<String, BufferedReader> result = new HashMap<>(9);
for (ZipEntry zipEntry = zipInputStream.getNextEntry(); zipEntry != null; zipEntry = zipInputStream.getNextEntry()) {
if (zipEntry.isDirectory()) {
continue;
}
result.put(zipEntry.getName(), new BufferedReader(new InputStreamReader(zipInputStream)));
}
return result;
}
同一类中的其他签名(调用相同的方法):
public static Map<String, BufferedReader> unzipFile(String zippedFile) throws IOException {
return unzipFile(new File(zippedFile));
}
public static Map<String, BufferedReader> unzipFile(File zippedFile) throws IOException {
return unzipFile(new FileInputStream(zippedFile));
}
我的想法是获取从此方法返回的Map
,并能够在我的代码上的其他地方读取它。问题是每次我在这个方法中调用result.get("filename").readLine()
循环之外的for
时,它返回null
。
这里有一个简单的单元测试方法 - 目前在最后一个Assert.assertNotNull
失败:
@Test
public void unzipFileTest() throws Exception {
Map<String, BufferedReader> result = FileHandlerUtility.unzipFile(TestUtil.FILE_SAMPLE_NAME);
Assert.assertNotNull(result);
Assert.assertEquals(result.size(), 9);
Assert.assertNotNull(result.get("Car.txt"));
Assert.assertNotNull(result.get("Client.txt"));
Assert.assertNotNull(result.get("Client.txt").readLine());
}
我想可能会有一些与创建的变量范围有关的东西,因为在调试时,我注意到我可以在for
循环中获取文件的内容。但是,我不想将此zip提取方法与获取内容并解析它的方法混合使用。此外,我不想将提取的文件保存到光盘并稍后重新打开它们。
所以,我怎么能用Map
填充这个BufferedReader
不会在readLine
调用上返回null?
每次迭代条目指针时,都会丢失条目指针,因为每个ZipInputStream对象只允许一个流。
try (ZipInputStream is = new ZipInputStream(Zippy.class.getResourceAsStream("file.zip"))) {
ZipEntry entry;
while ((entry = is.getNextEntry()) != null) {
if (!entry.isDirectory()) {
// do your logic here (get the entry name)
}
is.closeEntry();
}
}
closeEntry()
-Closes the current ZIP entry and positions the stream for reading the next entry.
getNextEntry() - Reads the next ZIP file entry and positions the stream at the beginning of the entry data.
基本上,您只能在任何给定时间打开一个条目。
因此,如果你想做你正在做的事情,你可以做的最好的事情是保存zip条目的名称并迭代ZipInputStream将指针移动到正确的位置,然后你可以使用ZipInputStream.read()来获取字节。
如果您可以使用ZipFile对象(而不是ZipInputStream),您可以执行您要完成的任务。
static void loadMap() throws IOException {
ZipFile file = new ZipFile("testzip.zip");
Enumeration<? extends ZipEntry> entries = file.entries();
while (entries.hasMoreElements()) {
ZipEntry entry = entries.nextElement();
if (!entry.isDirectory()) {
streamMap.put(entry.getName(), new BufferedReader(new InputStreamReader(file.getInputStream(entry))));
}
}
}
public static void main(String[] args) throws IOException {
loadMap();
Iterator<BufferedReader> iter = streamMap.values().iterator();
while (iter.hasNext()) {
BufferedReader reader = iter.next();
String line;
while ((line = reader.readLine()) != null) {
System.out.println(line);
}
}
}
OUTPUT:
file 2: First line!
file 2: Second Line!
file 1: First line!
file 1 : Second Line!
BUILD SUCCESSFUL (total time: 0 seconds)
您只需要记住保存zip文件引用并在完成后调用file.close();
。