gpt4 book ai didi

java - Files.walk(),计算总大小

转载 作者:IT老高 更新时间:2023-10-28 20:44:20 29 4
gpt4 key购买 nike

我正在尝试计算光盘上文件的大小。在 java-7 中,这可以使用 Files.walkFileTree 来完成。如我的回答 here 所示.

但是,如果我想使用 java-8 流来执行此操作,它将适用于某些文件夹,但不适用于所有文件夹。

public static void main(String[] args) throws IOException {
long size = Files.walk(Paths.get("c:/")).mapToLong(MyMain::count).sum();
System.out.println("size=" + size);
}

static long count(Path path) {
try {
return Files.size(path);
} catch (IOException | UncheckedIOException e) {
return 0;
}
}

上面的代码适用于路径 a:/files/ 但对于 c:/ 它将抛出异常

Exception in thread "main" java.io.UncheckedIOException: java.nio.file.AccessDeniedException: c:\$Recycle.Bin\S-1-5-20
at java.nio.file.FileTreeIterator.fetchNextIfNeeded(Unknown Source)
at java.nio.file.FileTreeIterator.hasNext(Unknown Source)
at java.util.Iterator.forEachRemaining(Unknown Source)
at java.util.Spliterators$IteratorSpliterator.forEachRemaining(Unknown Source)
at java.util.stream.AbstractPipeline.copyInto(Unknown Source)
at java.util.stream.AbstractPipeline.wrapAndCopyInto(Unknown Source)
at java.util.stream.ReduceOps$ReduceOp.evaluateSequential(Unknown Source)
at java.util.stream.AbstractPipeline.evaluate(Unknown Source)
at java.util.stream.LongPipeline.reduce(Unknown Source)
at java.util.stream.LongPipeline.sum(Unknown Source)
at MyMain.main(MyMain.java:16)

我了解它的来源以及如何使用 Files.walkFileTree API 来避免它。

但是如何使用 Files.walk() 来避免这个异常? API?

最佳答案

不,这个异常(exception)是无法避免的。

异常本身发生在 Files.walk() 的延迟提取中,因此为什么您没有及早看到它以及为什么没有办法规避它,请考虑以下代码:

long size = Files.walk(Paths.get("C://"))
.peek(System.out::println)
.mapToLong(this::count)
.sum();

在我的系统上,这将在我的计算机上打印:

C:\
C:\$Recycle.Bin
Exception in thread "main" java.io.UncheckedIOException: java.nio.file.AccessDeniedException: C:\$Recycle.Bin\S-1-5-18

当第三个文件的(主)线程抛出异常时,该线程上的所有进一步执行都会停止。

我认为这是设计失败,因为现在 Files.walk 绝对无法使用,因为您永远无法保证遍历目录时不会出错。

需要注意的重要一点是,stacktrace 包含 sum()reduce() 操作,这是因为路径是延迟加载的,所以在reduce() 点,大量流机制被调用(在堆栈跟踪中可见),然后它获取路径,此时 UnCheckedIOException 发生。

如果您让每个步行操作在它们自己的线程上执行,它可能可能被规避。但这不是你想做的事情。

另外,检查一个文件是否真的可以访问是毫无值(value)的(虽然在某种程度上很有用),因为你不能保证它在 1ms 之后仍然是可读的。

future 扩展

我相信它仍然可以修复,虽然我不知道 FileVisitOption 的具体工作原理。
目前有一个 FileVisitOption.FOLLOW_LINKS,如果它基于每个文件运行,那么我怀疑也可以添加一个 FileVisitOption.IGNORE_ON_IOEXCEPTION,但是我们无法正确注入(inject)那里的功能。

关于java - Files.walk(),计算总大小,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22867286/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com