Как вы динамически создаете класс с данным методом и телом метода в Ruby? - PullRequest
2 голосов
/ 24 ноября 2011

Как в Ruby определить метод

def make_class(method_name, method_body, s_value)

возвращая класс со следующей реализацией

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

    def to_s
       return s_value
    end
end

Если вы можете ссылаться на любые ресурсы, которые вы считаете полезными в базовом метапрограммировании на Ruby, это было бы замечательно.

Ответы [ 4 ]

5 голосов
/ 24 ноября 2011

Вы можете использовать что-то вроде этого:

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 ваш друг.

2 голосов
/ 24 ноября 2011
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

"Метапрограммирование Ruby: программа, подобная Ruby Pros" - очень хорошая книга для изучения метапрограммирования ruby.

1 голос
/ 25 ноября 2011

Самый простой способ, которым я могу придумать:

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

Редактировать: хорошо, я был слишком упрощен, здесь не рассматриваются аргументы дляметод.

0 голосов
/ 24 ноября 2011

я бы не советовал делать это, но вы могли бы ...

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
...