gpt4 book ai didi

java - 为什么 Java 7 Files.walkFileTree 在遇到远程驱动器上的 tar 文件时抛出异常

转载 作者:太空狗 更新时间:2023-10-29 22:35:50 24 4
gpt4 key购买 nike

我正在使用 Files.WalkFileTree() 来导航文件夹和计算音频文件,但是遇到 tar 文件时出现问题,它似乎将其视为我所期待的实际文件夹跳过它。

我看不到任何让我控制此行为的选项

代码:

package com.jthink.songkong.fileloader;


import com.jthink.songkong.cmdline.SongKong;
import com.jthink.songkong.ui.MainWindow;

import java.nio.file.*;
import java.nio.file.attribute.BasicFileAttributes;
import java.util.concurrent.Callable;
import java.util.logging.Level;

/**
* Count the number of files that can be loaded, for information purposes only
*/
public class CountFilesinFolder implements Callable<Boolean> {
public static class CountFiles
extends SimpleFileVisitor<Path> {
private int fileCount = 0;
private final PathMatcher matcher;

CountFiles(String pattern) {
matcher =
FileSystems.getDefault()
.getPathMatcher("regex:" + pattern);
}

/**
* Find Music file
*
* @param file
* @param attr
* @return
*/
@Override
public FileVisitResult visitFile(Path file,
BasicFileAttributes attr) {
Path name = file.getFileName();
if (name != null && matcher.matches(name)) {
fileCount++;
}
return FileVisitResult.CONTINUE;
}

public int getFileCount() {
return fileCount;
}
}


private Path scanDir;
public CountFilesinFolder(Path scanDir) {
this.scanDir = scanDir;
}

public Boolean call() {
CountFiles countFiles = null;
try {
countFiles = new CountFiles("^(?!._).*[.](?:mp3|mp4|m4p|m4b|m4a|ogg|flac|wma)$");
Files.walkFileTree(scanDir, countFiles);
}
catch (Exception e) {
MainWindow.logger.log(Level.SEVERE, "Unable to find file for deriving base folder", e);
}
MainWindow.logger.severe("Music File Count:"+countFiles.getFileCount());
SongKong.setMaxProgress(countFiles.getFileCount());
return true;
}
}

给出这个堆栈跟踪

java.nio.file.NoSuchFileException: Z:\Scratch\fred.tar
at sun.nio.fs.WindowsException.translateToIOException(WindowsException.java:79)
at sun.nio.fs.WindowsException.rethrowAsIOException(WindowsException.java:97)
at sun.nio.fs.WindowsException.rethrowAsIOException(WindowsException.java:102)
at sun.nio.fs.WindowsDirectoryStream.<init>(WindowsDirectoryStream.java:86)
at sun.nio.fs.WindowsFileSystemProvider.newDirectoryStream(WindowsFileSystemProvider.java:526)
at java.nio.file.Files.newDirectoryStream(Files.java:411)
at java.nio.file.FileTreeWalker.walk(FileTreeWalker.java:179)
at java.nio.file.FileTreeWalker.walk(FileTreeWalker.java:199)
at java.nio.file.FileTreeWalker.walk(FileTreeWalker.java:69)
at java.nio.file.Files.walkFileTree(Files.java:2591)
at java.nio.file.Files.walkFileTree(Files.java:2624)
at com.jthink.songkong.fileloader.CountFilesinFolder.call(CountFilesinFolder.java:68)
at com.jthink.songkong.fileloader.CountFilesinFolder.call(CountFilesinFolder.java:15)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:334)
at java.util.concurrent.FutureTask.run(FutureTask.java:166)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1110)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:603)
at java.lang.Thread.run(Thread.java:722)

但那是一个远程驱动器(nas 驱动器),我在本地驱动器上没有这样的错误

编辑根据我认为有效的以下答案实现了以下内容

    @Override
public FileVisitResult preVisitDirectory(Path dir,
BasicFileAttributes attrs)
throws IOException {
if(dir.endsWith(".tar"))
{
return FileVisitResult.SKIP_SUBTREE;
}
return super.preVisitDirectory(dir, attrs);
}

但我的测试有误,它实际上不起作用,因为 FileTreeWalker 中失败的代码是在预访问方法之前调用的

try {
DirectoryStream<Path> stream = null;
FileVisitResult result;

// open the directory
try {
stream = Files.newDirectoryStream(file);
} catch (IOException x) {
return visitor.visitFileFailed(file, x);
} catch (SecurityException x) {
// ignore, as per spec
return FileVisitResult.CONTINUE;
}

// the exception notified to the postVisitDirectory method
IOException ioe = null;

// invoke preVisitDirectory and then visit each entry
try {
result = visitor.preVisitDirectory(file, attrs);
if (result != FileVisitResult.CONTINUE) {
return result;
}

最佳答案

当前问题的解决方法:

但是执行一个 visitFileFailed 应该没问题。

public class MyFileVisitor extends SimpleFileVisitor<Path> {
@Override
public FileVisitResult visitFileFailed(Path file, IOException exc) throws IOException {
if (file.toString().endsWith(".tar")) {
return FileVisitResult.CONTINUE;
}
return super.visitFileFailed(file, exc);
}
}

更新:如果我们仔细观察,我们可以看到 walkFileTree使用 Files.readAttributes转向当前的提供者:WindowsFileSystemProvider.readAttributes判断路径是否为目录。

正如评论中提到的那样,我也不认为错误出在 Java 实现中,而是 OS-native-call返回错误的属性

如果您想为此做一个解决方法,一个选择是实现您自己的文件系统,透明地包装 WindowsFileSystem 实现,除了 readAttributes 返回 .tar-paths 作为文件而不是 dir。

关于java - 为什么 Java 7 Files.walkFileTree 在遇到远程驱动器上的 tar 文件时抛出异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14436032/

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