Ruby: define_method против def - PullRequest
       31

Ruby: define_method против def

65 голосов
/ 09 октября 2008

Как упражнение по программированию, я написал фрагмент Ruby, который создает класс, создает два объекта из этого класса, monkeypatches один объект и полагается на method_missing для monkeypatch другой.

Вот сделка. Это работает как задумано:

class Monkey

  def chatter
    puts "I am a chattering monkey!"
  end

  def method_missing(m)
    puts "No #{m}, so I'll make one..."
    def screech
      puts "This is the new screech."
    end
  end
end

m1 = Monkey.new
m2 = Monkey.new

m1.chatter
m2.chatter

def m1.screech
  puts "Aaaaaargh!"
end

m1.screech
m2.screech
m2.screech
m1.screech
m2.screech

Вы заметите, что у меня есть параметр для method_missing. Я сделал это, потому что надеялся использовать define_method для динамического создания отсутствующих методов с соответствующим именем. Тем не менее, это не работает. На самом деле, даже используя define_method со статическим именем, вот так:

def method_missing(m)
  puts "No #{m}, so I'll make one..."
  define_method(:screech) do
    puts "This is the new screech."
  end
end

Заканчивается со следующим результатом:

ArgumentError: wrong number of arguments (2 for 1)

method method_missing   in untitled document at line 9
method method_missing   in untitled document at line 9
at top level    in untitled document at line 26
Program exited.

Что делает сообщение об ошибке более удивительным, так это то, что у меня есть только один аргумент для method_missing ...

Ответы [ 3 ]

136 голосов
/ 09 октября 2008

define_method (приватный) метод объекта Класс . Вы звоните из экземпляра . Нет метода экземпляра с именем define_method, поэтому он возвращается к вашему method_missing, на этот раз с :define_method (имя отсутствующего метода) и :screech (единственный аргумент, который вы передали define_method).

Попробуйте вместо этого (чтобы определить новый метод для всех объектов Monkey):

def method_missing(m)
    puts "No #{m}, so I'll make one..."
    self.class.send(:define_method, :screech) do
      puts "This is the new screech."
    end
end

Или вот это (чтобы определить его только для объекта, к которому он обращен, используя «собственный класс» объекта):

def method_missing(m)
    puts "No #{m}, so I'll make one..."
    class << self
      define_method(:screech) do
        puts "This is the new screech."
      end
    end
end
4 голосов
/ 27 августа 2010
def method_missing(m)
    self.class.class_exec do
       define_method(:screech) {puts "This is the new screech."}
    end 
end

метод screech будет доступен для всех объектов Monkey.

4 голосов
/ 28 октября 2008

self.class.define_method (: screech) не работает, так как define_method является закрытым методом Вы можете сделать это

class << self
    public :define_method
end
def method_missing(m)
puts "No #{m}, so I'll make one..."
Monkey.define_method(:screech) do
  puts "This is the new screech."
end
...