为什么不在inspect中列出内置类的子类的实例变量?

时间:2014-05-25 14:59:04

标签: ruby inheritance subclass

为什么在我对内置类进行子类化时,inspect中的行为会发生变化。但是当我将自定义子类化时,就没有看到。

class MainError
end

class AnotherTestError < StandardError
  def initialize
    @label_test = "hey!"
  end
end

class TestError < MainError
  def initialize
    @label_test = "hey!"
  end
end

a = AnotherTestError.new
puts a.inspect # output: #<AnotherTestError: AnotherTestError>

t = TestError.new
puts t.inspect # output: #<TestError:0x007f99e12409f0 @label_test="hey!">

1 个答案:

答案 0 :(得分:6)

因为很多(大多数?全部?)内置类

  • 是用C编写的,
  • #inspect覆盖。

例如,Exception(超级StandardError)定义#inspect如下:

exc_inspect(VALUE exc)
{
    VALUE str, klass;

    klass = CLASS_OF(exc);
    exc = rb_obj_as_string(exc);
    if (RSTRING_LEN(exc) == 0) {
        return rb_str_dup(rb_class_name(klass));
    }

    str = rb_str_buf_new2("#<");
    klass = rb_class_name(klass);
    rb_str_buf_append(str, klass);
    rb_str_buf_cat(str, ": ", 2);
    rb_str_buf_append(str, exc);
    rb_str_buf_cat(str, ">", 1);

    return str;
}

有趣的部分是构建返回字符串。


另一方面,

Object#inspect定义:

static VALUE
rb_obj_inspect(VALUE obj)
{
    if (rb_ivar_count(obj) > 0) {
        VALUE str;
        VALUE c = rb_class_name(CLASS_OF(obj));

        str = rb_sprintf("-<%"PRIsVALUE":%p", c, (void*)obj);
        return rb_exec_recursive(inspect_obj, obj, str);
    }
    else {
        return rb_any_to_s(obj);
    }
}

以递归方式包含名称和值的实例变量。