您应该使用IO.inspect 而不是IO.puts 来打印值的内部表示:
iex> IO.puts [1, 2]
^A^B # prints "^A^B"
:ok # return value of IO.puts
iex> IO.inspect [1, 2]
[1, 2] # prints "[1, 2]"
[1, 2] # return value of IO.inspect
但是你可能仍然会遇到一些特殊值的问题:
iex> IO.inspect [97, 98]
'ab' # prints "'ab'"
'ab' # return value of IO.inspect
为了解释这种行为,我们需要了解字符串在 Elixir 中是如何工作的。有两种字符串。二进制字符串(双引号)和字符列表(单引号)。在内部,这些是由更简单的原语构建的。二进制字符串由二进制构建,字符列表由列表构建:
iex> "ab" # a binary string
"ab"
iex> <<97, 98>> # the same string in binary syntax
"ab"
iex> 'ab' # a character list
'ab'
iex> [97, 98] # the same character list in list syntax
'ab'
起初可能看起来令人困惑,但这是由于两件事造成的。首先,没有内置的字符串类型,因为我们已经看到字符串是从其他原始类型构建的。其次,Elixir 中没有用户定义的类型。所以当 Elixir 看到一个只有整数的列表时,为了方便起见,它会尝试将其打印为字符串。然而,在底层它仍然只是一个整数列表。
在上面的示例中,97 和 98 表示字符 a 和 b 的 unicode 代码点,因此即使使用 IO.inspect,它们也会显示为字符串。
现在您可以看到为什么它在您的示例中打印 ^A^B - 这些只是 control characters from the ASCII encoding,恰好由代码点 1 和 2 表示。
但是,您可以传递选项 char_lists: :as_lists 以打印原始列表,而无需尝试进行此类转换:
iex> IO.inspect [97, 98], char_lists: :as_lists
[97, 98] # prints '[97, 98]'
'ab' # Iex still shows the return value as 'ab'
如果您打开 iex 并输入 h Inspect.Opts,您会看到 Elixir 也对其他值执行这种操作,特别是结构和二进制文件。