我认为执行puts #{a}
会产生与puts a
相同的输出,但发现情况并非如此。考虑:
irb(main):001:0> a = [1,2]
=> [1, 2]
irb(main):002:0> puts a
1
2
=> nil
irb(main):003:0> puts "#{a}"
12
=> nil
irb(main):004:0>
在上面的例子中它并不重要,但是当我想在一行上打印多个变量时可能很重要,例如(psudocode):
puts "There are #{a.size} items in the whitelist: #{a}"
为什么这里的输出不同?他们实际上做了不同的事情,还是有不同的语义?
答案 0 :(得分:6)
那是因为"#{a}"
调用表达式上的#to_s
方法。
所以:
puts a # equivalent to, well, puts a
puts "#{a}" # equivalent to the next line
puts a.to_s
更新
详细说明,puts
最终会调用#to_s
,但它会在实际输出前添加逻辑,包括对数组的特殊处理。恰好Array#to_s
不使用相同的算法。 (See puts
docs here.)这正是它的作用......
rb_io_puts(int argc, VALUE *argv, VALUE out)
{
int i;
VALUE line;
/* if no argument given, print newline. */
if (argc == 0) {
rb_io_write(out, rb_default_rs);
return Qnil;
}
for (i=0; i<argc; i++) {
if (TYPE(argv[i]) == T_STRING) {
line = argv[i];
goto string;
}
line = rb_check_array_type(argv[i]);
if (!NIL_P(line)) {
rb_exec_recursive(io_puts_ary, line, out);
continue;
}
line = rb_obj_as_string(argv[i]);
string:
rb_io_write(out, line);
if (RSTRING_LEN(line) == 0 ||
!str_end_with_asciichar(line, '\n')) {
rb_io_write(out, rb_default_rs);
}
}
return Qnil;
}