How to use Files.walk()… to get a graph of files based on conditions?

前端 未结 3 1768
名媛妹妹
名媛妹妹 2021-02-01 04:24

I have the following directory structures:

/path/to/stuff/org/foo/bar/
/path/to/stuff/org/foo/bar/1.2.3/
/path/to/stuff/org/foo/bar/1.2.3/myfile.ext
/path/to/stu         


        
相关标签:
3条回答
  • 2021-02-01 05:02

    You are invoking the method printIfArtifactVersionDirectory for all visited directories. I did a little change to make it obvious:

    static void printIfArtifactVersionDirectory(Path path) {
        System.out.println("--- " + path);
        ...
    }
    

    With that additional output you will get:

    --- C:\Projects\stuff
    --- C:\Projects\stuff\org
    --- C:\Projects\stuff\org\foo
    --- C:\Projects\stuff\org\foo\bar
    --- C:\Projects\stuff\org\foo\bar\1.2.3
    C:\Projects\stuff\org\foo\bar
    --- C:\Projects\stuff\org\foo\bar\1.2.4
    C:\Projects\stuff\org\foo\bar
    --- C:\Projects\stuff\org\foo\bar\blah
    --- C:\Projects\stuff\org\foo\bar\blah\2.1
    C:\Projects\stuff\org\foo\bar\blah
    --- C:\Projects\stuff\org\foo\bar\blah\2.2
    C:\Projects\stuff\org\foo\bar\blah

    So you get the output as often as you have artifact version directories. If you want to remember that you already did the output for one directory, you must make store this information somewhere. One quick implementation could be:

    static class Foo {
        private static final Set<Path> visited = new HashSet<>();
    
        static void printIfArtifactVersionDirectory(Path path) {
            ...
            Path parent = path.getParent();
            if (!filePaths.isEmpty() && !visited.contains(parent)) {
                visited.add(parent);
                System.out.println(parent);
            }
        }
    }
    

    With this you get the expected output:

    C:\Projects\stuff\org\foo\bar
    C:\Projects\stuff\org\foo\bar\blah

    A better solution would be to use the set for storing the visited parents and only print them after visiting them all:

    static class PathStore {
        private final Set<Path> store = new HashSet<>();
    
        void visit(Path path) {
            File f = path.toAbsolutePath().toFile();
            List<String> filePaths = Arrays.asList(f.list(new MyExtFilenameFilter()));
            if (!filePaths.isEmpty()) {
                store.add(path.getParent());
            }
        }
    
        void print() {
            store.forEach(System.out::println);
        }
    }
    

    Usage:

    PathStore pathStore = new PathStore();
    Files.walk(Paths.get("/path/to/stuff/"))
            .filter(Files::isDirectory)
            .forEach(pathStore::visit);
    pathStore.print();
    
    0 讨论(0)
  • 2021-02-01 05:05

    Adding to @a-better-oliver's answer, here is what you can do if your action declares IOExceptions.

    You can treat the filtered stream as an Iterable, and then do your action in a regular for-each loop. This way, you don't have to handle exceptions inside a lambda.

    try (Stream<Path> pathStream = Files
            .walk(Paths.get("/path/to/stuff/"))
            .filter(p -> p.toString().endsWith(".ext"))
            .map(p -> p.getParent().getParent())
            .distinct()) {
    
        for (Path file : (Iterable<Path>) pathStream::iterator) {
            // something that throws IOException
            Files.copy(file, System.out);
        }
    }
    

    Found that trick here: https://stackoverflow.com/a/32668807/1207791

    0 讨论(0)
  • 2021-02-01 05:17
    Files.walk(Paths.get("/path/to/stuff/"))
         .filter(p -> p.toString().endsWith(".ext"))
         .map(p -> p.getParent().getParent())
         .distinct()
         .forEach(System.out::println);
    

    This filters all files that have the extension and gets the parent path of their directory. distinct ensures that every path is used only once.

    0 讨论(0)
提交回复
热议问题