一个使用method_missing()容易引发的bug

class Roulette
    def method_missing(name,*args)
        person = name.to_s.capitalize
        3.times do
            number = rand(10) + 1
            puts "#{number} = ..."
        end
        puts "#{person} got a #{number}"
    end
end
number_of = Roulette.new
puts number_of.bob
puts number_of.frank

      Ruby元编程里搬过来的例子.这个程序里,number是定义在传给times方法的那个块中,在第八行引用的时候,已经超出了它的作用域.Ruby无从得知此处的number应该是一个变量.默认情况下,它把number当成一个在self上省略了括号的方法调用.
  通常情况下,会抛出一个明显的NoMethodError错误,但是在这个程序里,已经自己定义了一个method_missing()方法,所以number()方法的调用最终都会来到这里,因此会陷入死循环,直到调用堆栈溢出为止.
  这是使用幽灵方法的时候会经常出现的问题:由于调用未定义的方法会导致调用method_missing()方法,对象可能会因此接受一个直接的错误方法调用(不如输错了方法名).
  为了避免这样的困扰,应该仅在必要时才使用幽灵方法.碰到不知道如何处理的方法时,记得回到Kernel#method_missing()方法.

  修改后的方法:

class Roulette
    def method_missing(name,*args)
        person = name.to_s.capitalize
        super unless %w[Bob Frank Bill].include?person
        number = 0
        3.times do
            number = rand(10) + 1
            puts "#{number} = ..."
        end
        puts "#{person} got a #{number}"
    end
end
number_of = Roulette.new
puts number_of.bob
puts number_of.frank

 

posted on 2013-12-29 13:29  秋叶leaf  阅读(173)  评论(0编辑  收藏  举报