ruby元编程之method_missing的一个使用细节
我们知道顶级域,定义域的self是啥?
putsself #main putsself.class#Object
我们知道当一个方法被调用的时候,如果没有对象接受,默认就是self,如:
deftell_me_who putsself end tell_me_who #main
方法调用是这样的步骤,先查找当前对象的所在类的实例方法存在方法与否,如果存在,调用方法,如果不存在则查看superclass,直到BasicObject都没找到对于方法的话,就会调用Kernel的method_missing()方法,并且报错,如
Error:test.rb:8:undefine:undefinedlocalvariableormethod`ask'formain:Object(NameError)
注意报错的信息,我们可以发现,当我们调用一个不存在的变量的时候,也是会追溯到Kernel的method_missing方法的,这里要注意咯。
验证:
putsself #main putsself.class#Object defself.method_missing(name,*arg) puts"#{name}isnotexist!" end putsask #askisnotexist!
一个案例导致BUG:
defself.method_missing(name,*arg) 1.timesdo putsmethod_name=name end puts"#{method_name}isnotexist!" end
ask#变量或者方法
意图:让任何未定义的变量或者方法,都打印一次
可是,这是一个死循环?看出问题了吗
ask被执行,可是没有定义ask,就会转到method_missing,
method_name在times的block中,出了作用域了,所以又会执行method_missing,变成了死循环。