为什么我不能将匿名函数应用于列表?

Sol*_*Sol 2 functional-programming elixir

所以我正在尝试学习 Elixir(我有 F# 和 Haskell 的背景),但我很难理解我的代码中发生的事情:

fizz_buzz = fn
  (0, 0, _) -> "FizzBuzz"
  (0, _, _) -> "Fizz"
  (_, 0, _) -> "Buzz"
  (_, _, c) -> c
end

fizz_buzz_rem = fn n -> fizz_buzz.(rem(n, 3), rem(n, 5), n) end


# This works
IO.puts(fizz_buzz_rem.(10))
IO.puts(fizz_buzz_rem.(11))
IO.puts(fizz_buzz_rem.(12))
IO.puts(fizz_buzz_rem.(13))
IO.puts(fizz_buzz_rem.(14))
IO.puts(fizz_buzz_rem.(15))
IO.puts(fizz_buzz_rem.(16))
IO.puts(fizz_buzz_rem.(17))

IO.puts("----------------")

inputs =
  10..17
  |> Enum.to_list

# Doesn't work
inputs
|> Enum.map(fizz_buzz_rem)
|> IO.puts

IO.puts("----------------")

# Doesn't work
inputs
|> Enum.map(fn n -> fizz_buzz.(rem(n, 3), rem(n, 5), n) end)
|> IO.puts

IO.puts("----------------")

manual_inputs = [10, 11, 12, 13, 14, 15, 16, 17]

# Doesn't work
manual_inputs
|> Enum.map(fizz_buzz_rem)
|> IO.puts

IO.puts("----------------")

# Doesn't work
manual_inputs
|> Enum.map(fn n -> fizz_buzz.(rem(n, 3), rem(n, 5), n) end)
|> IO.puts


IO.puts("----------------")

# The idiotic way (that doesn't work)

result = [
  fizz_buzz_rem.(10),
  fizz_buzz_rem.(11),
  fizz_buzz_rem.(12),
  fizz_buzz_rem.(13),
  fizz_buzz_rem.(14),
  fizz_buzz_rem.(15),
  fizz_buzz_rem.(16),
  fizz_buzz_rem.(17),
]

IO.puts result

# ???????????
Run Code Online (Sandbox Code Playgroud)

当我运行时elixir ex_02.exs,输出是:

Buzz
FizzBuzz
Buzz
11
Fizz
13
14
FizzBuzz
16
17
----------------
Buzz
FizzBuzz
----------------
Buzz
FizzBuzz
----------------
Buzz
FizzBuzz
----------------
Buzz
FizzBuzz
Run Code Online (Sandbox Code Playgroud)

因此,正如您所看到的,当我将匿名函数单独应用于每个值时,当我尝试使用范围、映射甚至手动将函数应用于列表的每个元素时,我得到了正确的答案,但最终得到了错误的结果。

将匿名函数应用到 elixir 中的列表有什么问题?

leg*_*cia 7

如果你使用IO.inspect而不是IO.puts,你可以看到发生了什么:

["Buzz", 11, "Fizz", 13, 14, "FizzBuzz", 16, 17]
Run Code Online (Sandbox Code Playgroud)

您的 fizzbuzz 函数返回字符串或整数,具体取决于输入。 IO.puts根据它们是否在列表中,以不同的方式对待整数:

iex(1)> IO.puts(65)
65
:ok
iex(2)> IO.puts([65])
A
:ok
Run Code Online (Sandbox Code Playgroud)

因此,在您的代码中,IO.puts实际上打印了与整数 11、13、14、16 和 17 相对应的控制代码。在我的终端中,它显示为:

Buzz^KFizz^M^NFizzBuzz^P^Q
Run Code Online (Sandbox Code Playgroud)

您可以通过使您的函数始终返回字符串来解决此问题:

fizz_buzz = fn
  (0, 0, _) -> "FizzBuzz"
  (0, _, _) -> "Fizz"
  (_, 0, _) -> "Buzz"
  (_, _, c) -> "#{c}"
end
Run Code Online (Sandbox Code Playgroud)

  • 这是意外地雷的另一个受害者,当 Elixir 将整数解释为 ASCII 代码点时,这种地雷可能会爆炸——在这种情况下,所有这些整数都映射到各种空白或控制字符,因此你看不到发生了什么。查看类似的问题/答案,例如/sf/ask/4605639321/#65797867 (3认同)