Я изучаю метапрограммирование в Ruby и просто пытаюсь определить отсутствующие методы с помощью method_missing и define_method. Я получаю неожиданное поведение, и мне интересно, может ли кто-нибудь объяснить это. Вот мой класс:
class X
def method_missing(m, *args, &block)
puts "method #{m} not found. Defining it."
self.class.send :define_method, m do
puts "hi from method #{m}"
end
puts "defined method #{m}"
end
end
Теперь этот код:
x = X.new
x.some_method
puts
x.some_method
puts
puts x
Производит вывод:
method some_method not found. Defining it.
defined method some_method
hi from method some_method
method to_ary not found. Defining it.
defined method to_ary
#<X:0x007fcbc38e5030>
Чего я не понимаю, так это последней части: почему Ruby вызывает to_ary в вызове puts? Зачем Ruby пытаться преобразовать мой объект в массив только для того, чтобы распечатать его?
Я погуглил и нашел эти связанные ссылки:
- http://tenderlovemaking.com/2011/06/28/til-its-ok-to-return-nil-from-to_ary/
- http://yehudakatz.com/2010/01/02/the-craziest-fing-bug-ive-ever-seen/
В них также говорится о подводных камнях method_missing и to_ary, но не конкретно о том, почему puts вызывает to_ary.
Я также должен упомянуть, что поведение не меняется, когда я определяю to_s, например.
def to_s
"I'm an instance of X"
end
Выход «помещает x» тогда:
method to_ary not found. Defining it.
defined method to_ary
I'm an instance of X