あなたはどのようにして、Rubyで特定のメソッドとメソッド本体を持つクラスを動的に作成しますか?

In ruby how would you define a method

def make_class(method_name、method_body、s_value)

returning a class with the following implementation

class Anonymous
    def method_name(args)
        method_body(args)
    end

    def to_s
       return s_value
    end
end

基本的なRubyメタプログラミングに役立つと思われるリソースにリンクすることができれば、それも素晴らしいでしょう。

2

4 答え

あなたはそのようなsmthを使うことができます:

def make_class(s_value, method_name, &method_body)    
  Class.new do   
    define_method method_name, method_body    

    define_method :to_s do    
      s_value    
    end    
  end
end    

klass = make_class 'foo instance', :foo do |*args|    
  "called foo with #{args.inspect}"    
end    

k = klass.new     
puts k.to_s                 # => foo instance
puts k.foo [1, 2], 'hello'  # => called foo with [[1, 2], "hello"]

この場合、メソッドの本体をブロックとして渡すことになっています( | * args | をメソッドのパラメータとして使用する引数のリストで置き換えることができます)。 method_body をブロックではなく文字列として渡したい場合、 eval はあなたの友人です。

5
追加された
ちょうどこのスニペットを実行し、それは動作します。結局のところ、メタプログラミングはあまり面倒ではありません。ありがとう!
追加された 著者 Garrett Hall,
def make_class(method_name, method_body, s_value)
  Class.new {
    define_method method_name do |*args|
      eval(method_body)
    end

    define_method :to_s do
      s_value
    end
  }
end

"Metaprogramming Ruby: Program Like the ruby Pros" is a very good book to learn about ruby metaprogramming.

2
追加された
本をありがとう、私はそれをチェックします。
追加された 著者 Garrett Hall,

私が考えることができる最も簡単な方法:

def make_class(method_name, method_body, s_value)
  klass = Class.new
  klass.class_eval "def #{method_name} ; #{method_body} ; end"
  klass.class_eval "def to_s ; #{s_value} ; end"
  klass
end

使用法:

>> Anonymous = make_class(:foobar, "puts 'foo'", 23)
=> Anonymous
>> a = Anonymous.new
=> 23
>> a.foobar
foo
=> nil
>> a.to_s
=> 23

編集:大丈夫、私はここで少し単純すぎた、これはメソッドの引数を処理しません。

1
追加された

私はこれをすることをお勧めしませんが、あなたができる...

def make_class(method_name, method_body, s_value)
  eval("
  class Anonymous
      def #{method_name}(args)
          #{method_body}(args)
      end

      def to_s
         return '#{s_value}'
      end
  end
  ")
end

make_class(:bla, :puts, 'bla')
Anonymous.new.bla('moin')
puts Anonymous.new.to_s

返す

moin
bla
0
追加された