30

Trying to use define_method inside initialize but getting undefined_method define_method. What am I doing wrong?

class C
  def initialize(n)    
    define_method ("#{n}") { puts "some method #{n}" }    
  end
end

C.new("abc") #=> NoMethodError: undefined method `define_method' for #<C:0x2efae80>
mu is too short
  • 426,620
  • 70
  • 833
  • 800
Bala
  • 11,068
  • 19
  • 67
  • 120

3 Answers3

47

I suspect that you're looking for define_singleton_method:

define_singleton_method(symbol, method) → new_method
define_singleton_method(symbol) { block } → proc

Defines a singleton method in the receiver. The method parameter can be a Proc, a Method or an UnboundMethod object. If a block is specified, it is used as the method body.

If you use define_method on self.class, you'll create the new method as an instance method on the whole class so it will be available as a method on all instances of the class.

You'd use define_singleton_method like this:

class C
  def initialize(s)    
    define_singleton_method(s) { puts "some method #{s}" }    
  end
end

And then:

a = C.new('a')
b = C.new('b')
a.a # puts 'some method a'
a.b # NoMethodError
b.a # NoMethodError
b.b # puts 'some method b'

If your initialize did:

self.class.send(:define_method,n) { puts "some method #{n}" }    

then you'd get:

a.a # puts 'some method a'
a.b # puts 'some method b'
b.a # puts 'some method a'
b.b # puts 'some method b'

and that's probably not what you're looking for. Creating a new instance and having the entire class change as a result is rather odd.

mu is too short
  • 426,620
  • 70
  • 833
  • 800
28

Do as below :

class C
  def initialize(n)    
    self.class.send(:define_method,n) { puts "some method #{n}" }    
  end
end

ob = C.new("abc")
ob.abc
# >> some method abc

Module#define_method is a private method and also a class method.Your one didn't work,as you tried to call it on the instance of C.You have to call it on C,using #send in your case.

Arup Rakshit
  • 116,827
  • 30
  • 260
  • 317
  • 1
    It works. Could you explain why my earlier version is incorrect? – Bala Oct 14 '13 at 20:14
  • 2
    @Bala: Note that `o = C.new('pancakes')` will leave you with `ob.abc`, `ob.pancakes`, `o.abc`, and `o.pancakes` as valid method calls. Is everyone sure that this is the intent? – mu is too short Oct 14 '13 at 20:21
  • 1
    @muistooshort: Interesting comment. Although there was no intent, now I am thinking how to make it object specific (singleton). – Bala Oct 14 '13 at 20:25
0

You were almost there. Just point to the class with self.class, don't even need to use :send:

class C
  def initialize(n)    
    self.class.define_method ("#{n}") { puts "some method #{n}" }    
  end
end
ob = C.new('new_method')
ob2 = C.new('new_method2')
# Here ob and ob2 will have access to new_method and new_method2 methods

You can also use it with :method_missing to teach your class new methods like this:

class Apprentice
  def method_missing(new_method)
    puts "I don't know this method... let me learn it :)"
    self.class.define_method(new_method) do
      return "This is a method I already learned from you: #{new_method}"
    end
  end
end
ap = Apprentice.new
ap.read
=> "I don't know this method... let me learn it :)"
ap.read
=> "This is a method I already learned from you: read"
Sergio Gonzalez
  • 1,851
  • 1
  • 12
  • 12