2

我认为这样做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}"

为什么这里的输出不同?他们实际上做了不同的事情,还是有不同的语义?

4

1 回答 1

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不使用相同的算法。(请参阅puts此处的文档。)这正是它的作用......

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;
}
于 2012-11-09T15:45:17.670 回答