Bala
Bala

Reputation: 11244

How to use define_method inside initialize()

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>

Upvotes: 30

Views: 13436

Answers (3)

Sergio Gonzalez
Sergio Gonzalez

Reputation: 2190

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"

Upvotes: 0

mu is too short
mu is too short

Reputation: 434665

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.

Upvotes: 47

Arup Rakshit
Arup Rakshit

Reputation: 118271

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.

Upvotes: 28

Related Questions