Я хотел бы передать аргумент методу, определяемому с помощью define_method, как бы я это сделал?
Как передать аргументы define_method?
Ответ 1
Блок, который вы передаете define_method, может содержать некоторые параметры. То, как ваш определенный метод принимает аргументы. Когда вы определяете метод, вы действительно просто прозвали блок и сохранили ссылку на него в классе. Параметры поставляются с блоком. Итак:
define_method(:say_hi) { |other| puts "Hi, " + other }
Ответ 2
... и если вы хотите дополнительные параметры
 class Bar
   define_method(:foo) do |arg=nil|                  
     arg                                                                                          
   end   
 end
 a = Bar.new
 a.foo
 #=> nil
 a.foo 1
 # => 1
... столько аргументов, сколько хотите
 class Bar
   define_method(:foo) do |*arg|                  
     arg                                                                                          
   end   
 end
 a = Bar.new
 a.foo
 #=> []
 a.foo 1
 # => [1]
 a.foo 1, 2 , 'AAA'
 # => [1, 2, 'AAA']
... комбинация
 class Bar
   define_method(:foo) do |bubla,*arg|
     p bubla                  
     p arg                                                                                          
   end   
 end
 a = Bar.new
 a.foo
 #=> wrong number of arguments (0 for 1)
 a.foo 1
 # 1
 # []
 a.foo 1, 2 ,3 ,4
 # 1
 # [2,3,4]
... все из них
 class Bar
   define_method(:foo) do |variable1, variable2,*arg, &block|  
     p  variable1     
     p  variable2
     p  arg
     p  block.inspect                                                                              
   end   
 end
 a = Bar.new      
 a.foo :one, 'two', :three, 4, 5 do
   'six'
 end
Обновление
Ruby 2.0 представил двойной splat ** (две звезды), который (цитирую):
Ruby 2.0 представил аргументы ключевых слов и ** действует как *, но для аргументов ключевого слова. Он возвращает Hash с парами ключ/значение.
... и, конечно, вы можете использовать его и в методе определения:)
 class Bar 
   define_method(:foo) do |variable1, variable2,*arg,**options, &block|
     p  variable1
     p  variable2
     p  arg
     p  options
     p  block.inspect
   end 
 end 
 a = Bar.new
 a.foo :one, 'two', :three, 4, 5, ruby: 'is awesome', foo: :bar do
   'six'
 end
# :one
# "two"
# [:three, 4, 5]
# {:ruby=>"is awesome", :foo=>:bar}
Пример именованных атрибутов:
 class Bar
   define_method(:foo) do |variable1, color: 'blue', **other_options, &block|
     p  variable1
     p  color
     p  other_options
     p  block.inspect
   end
 end
 a = Bar.new
 a.foo :one, color: 'red', ruby: 'is awesome', foo: :bar do
   'six'
 end
# :one
# "red"
# {:ruby=>"is awesome", :foo=>:bar}
Я пытался создать пример с аргументом ключевого слова, splat и double splat в одном:
 define_method(:foo) do |variable1, variable2,*arg, i_will_not: 'work', **options, &block|
    # ...
или
 define_method(:foo) do |variable1, variable2, i_will_not: 'work', *arg, **options, &block|
    # ...
... но это не сработает, похоже, есть ограничение. Когда вы думаете об этом, имеет смысл, поскольку оператор splat "захватывает все остальные аргументы", а double splat "захватывает все оставшиеся аргументы ключевых слов", поэтому их смешивание приведет к поломке ожидаемой логики. (У меня нет никакой ссылки, чтобы доказать эту точку doh!)
Ответ 3
В дополнение к ответу Кевина Коннера: аргументы блока не поддерживают ту же семантику, что и аргументы метода. Вы не можете определять аргументы по умолчанию или блокировать аргументы.
Это только исправлено в Ruby 1.9 с новым альтернативным синтаксисом "stabby lambda", который поддерживает полную семантику аргументов метода.
Пример:
# Works
def meth(default = :foo, *splat, &block) puts 'Bar'; end
# Doesn't work
define_method :meth { |default = :foo, *splat, &block| puts 'Bar' }
# This works in Ruby 1.9 (modulo typos, I don't actually have it installed)
define_method :meth, ->(default = :foo, *splat, &block) { puts 'Bar' }
Ответ 4
В версии 2.2 вы можете использовать аргументы ключевого слова: https://robots.thoughtbot.com/ruby-2-keyword-arguments
define_method(:method) do |refresh: false|
  ..........
end
