为什么在我对内置类进行子类化时,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!">
答案 0 :(得分:6)
因为很多(大多数?全部?)内置类
#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;
}
有趣的部分是构建返回字符串。
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);
}
}
以递归方式包含名称和值的实例变量。