【问题标题】:How do you filter hidden files using DirectoryStream.Filter如何使用 DirectoryStream.Filter 过滤隐藏文件
【发布时间】:2020-05-13 21:19:47
【问题描述】:

我正在尝试使用 NIO 类过滤隐藏文件。

当我在 Windows 10 上运行附加代码时,我得到以下输出:

Files:
        c:\Documents and Settings
        c:\PerfLogs
        c:\Program Files
        c:\Program Files (x86)
        c:\Users
        c:\Windows

Paths:
        c:\$Recycle.Bin
        c:\Config.Msi
        c:\Documents and Settings
        c:\Intel
        c:\IntelOptaneData
        c:\OEM
        c:\OneDriveTemp
        c:\PerfLogs
        c:\Program Files
        c:\Program Files (x86)
        c:\ProgramData
        c:\Recovery
        c:\System Volume Information
        c:\Users
        c:\Windows

文件下显示的列表(使用旧的File.listFiles(FileFilter) 方法)是我在 Windows 文件资源管理器中看到的列表,也是我期望看到的列表(文档和设置除外,我知道如何解决该问题)

  1. 为什么 NIO 方法过滤隐藏文件的方式不一样?
  2. 如何使 NIO 过滤保持相同?

这里是测试代码:

import java.io.*;
import java.nio.file.*;

public class ListFilesNIO
{
    public static void main(String[] args) throws Exception
    {
        String directory = "c:\\";

        //  Use old File I/O

        FileFilter fileFilter = new FileFilter()
        {
            @Override
            public boolean accept(File entry)
            {
                if (entry.isHidden()) return false;

                return true;
            }
        };

        System.out.println("Files:");
        File[] files = new File( directory ).listFiles( fileFilter );

        for (File file : files)
        {
            System.out.println( "\t" + file );
        }

        //  Use NIO

        DirectoryStream.Filter<Path> pathFilter = new DirectoryStream.Filter<Path>()
        {
            @Override
            public boolean accept(Path entry) throws IOException
            {
                if (Files.isHidden( entry )) return false;

                return true;
            }
        };

        System.out.println();
        System.out.println("Paths:");
        DirectoryStream<Path> paths = Files.newDirectoryStream(Paths.get( directory ), pathFilter);

        for (Path path : paths)
        {
            System.out.println( "\t" + path );
        }
    }
}

注意:当我在没有过滤器的情况下运行代码时,在这两种情况下都会显示 18 个文件。所以第一种方法是过滤12个隐藏文件,第二种方法只过滤3个文件。

【问题讨论】:

标签: java nio


【解决方案1】:

这不是错误,而是自 jdk7 以来已知的功能(!),Windows 隐藏目录未被检测为隐藏,请参阅此 bug 和此 one(修复 jdk13)。

作为一种解决方法,您可以这样做:

import java.nio.file.attribute.DosFileAttributes;
...
DirectoryStream.Filter<Path> pathFilter = new DirectoryStream.Filter<Path>()
        {
            @Override
            public boolean accept(Path entry) throws IOException
            {
               DosFileAttributes attr = Files.readAttributes(entry, DosFileAttributes.class);
               return !attr.isHidden();
            }
        };

【讨论】:

    【解决方案2】:

    我最终使用了:

    DirectoryStream.Filter<Path> pathFilter = new DirectoryStream.Filter<Path>()
    {
        @Override
        public boolean accept(Path entry) throws IOException
        {
            DosFileAttributes attr = Files.readAttributes(entry, DosFileAttributes.class, LinkOption.NOFOLLOW_LINKS);
    
            return !attr.isHidden();
        }
    };
    

    正如我在问题中提到的,我还希望隐藏 Documents and Settings

    Documents and Settings 是指向C:\Users 的链接。

    Files.readAttributes(…) 方法的默认实现是跟随链接。所以我猜是因为c:\Users 目录没有隐藏,所以Documents and Settings 也被认为没有隐藏。

    通过使用LinkOption.NOFOLLOW_LINKS,它被认为是隐藏的,这就是我想要的。

    【讨论】:

      猜你喜欢
      • 2020-11-15
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2011-12-08
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多