长生不老药 Enum.map 通过管道传输到 Enum.filter 无效

Elixir Enum.map piped to Enum.filter not working

我正在尝试编写一个工具,它将从 File.ls 中获取 files/folders 的列表,向它们添加完整路径,并根据它们是否是目录来过滤它们。

这个有效:

directory_contents = File.ls!(directory_path)

contents_with_full_paths = Enum.map directory_contents, fn(item) -> Path.join(directory_path, item) end
only_dirs = Enum.filter contents_with_full_paths, fn(item) -> File.dir?(item) end

这不是:

directory_contents = File.ls!(directory_path)

directory_contents
|> Enum.map fn(item) -> Path.join(directory_path, item) end
|> Enum.filter fn(item) -> File.dir?(item) end

它抛出

** (Protocol.UndefinedError) protocol Enumerable not implemented for #Function<1.10194857/1 in RepoFinder.subdirectories_in/1>, only anonymous functions of arity 2 are enumerable. This protocol is implemented for: Date.Range, File.Stream, Function, GenEvent.Stream, HashDict, HashSet, IO.Stream, List, Map, MapSet, Range, Stream
    (elixir) lib/enum.ex:3213: Enumerable.Function.reduce/3
    (elixir) lib/enum.ex:1847: Enum.filter/2

这是为什么?这两个实现的工作方式不应该基本相同吗?

编译代码时,编译器发出了详细的警告,解释了这段代码的错误:

warning: parentheses are required when piping into a function call. For example:

foo 1 |> bar 2 |> baz 3

is ambiguous and should be written as

foo(1) |> bar(2) |> baz(3)

Ambiguous pipe found at:
    /path/to/file.ex:line

也就是说,对于管道运算符,必​​须在函数调用的参数周围使用括号:

directory_contents
|> Enum.map(fn(item) -> Path.join(directory_path, item) end)
|> Enum.filter(&File.dir?/1)

否则,编译器会使用错误的运算符优先级。