Как вы передаете аргументы для define_method?

155

Я хотел бы передать аргумент (ы) методу, определяемому с помощью define_method, как бы я это сделал?

Sixty4Bit
источник

Ответы:

198

Блок, который вы передаете define_method, может содержать некоторые параметры. Вот как ваш определенный метод принимает аргументы. Когда вы определяете метод, вы на самом деле просто называете блок и сохраняете ссылку на него в классе. Параметры поставляются с блоком. Так:

define_method(:say_hi) { |other| puts "Hi, " + other }
Кевин Коннер
источник
Ну, это просто чистая красота. Отличная работа, Кевин Костнер.
Дарт Эгрегиус
90

... и если вы хотите дополнительные параметры

 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 появился двойной сплат **(две звезды), который ( я цитирую ) делает:

В Ruby 2.0 введены ключевые аргументы, и ** действует как *, но для ключевых слов. Возвращает хэш с парами ключ / значение.

... и, конечно, вы можете использовать его и в методе define :)

 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}

Я пытался создать пример с аргументом ключевого слова, сплатом и двойным сплатом в одном:

 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 «захватывает все оставшиеся аргументы ключевого слова», поэтому их смешивание нарушит ожидаемую логику. (У меня нет никаких ссылок, чтобы доказать эту точку зрения дох!)

обновление 2018 августа:

Сводная статья: https://blog.eq8.eu/til/metaprogramming-ruby-examples.html

equivalent8
источник
Интересно - специально 4-й блок: он работал на 1.8.7! Первый блок не работал в 1.8.7, а второй блок имеет опечатку (должно быть a.foo 1вместо foo 1). Спасибо!
Sony Santos
1
спасибо за отзыв, опечатка была исправлена, ... На ruby ​​1.9.3 и 1.9.2 все примеры работают, и я уверен, что на 1.9.1 тоже (но не пробовал)
эквивалент 8
Я объединил этот ответ с принятым ответом на stackoverflow.com/questions/4470108/… чтобы понять, как перезаписать (не переопределить) метод во время выполнения, который принимает необязательные аргументы и блок, и при этом может вызывать исходный метод с аргументами и заблокировать. Ах, рубин. В частности, мне нужно было переписать Savon :: Client.request в моей dev-среде для одного API-вызова хоста, к которому я могу получить доступ только в производственной среде. Ура!
pduey
59

В дополнение к ответу Кевина Коннера: блочные аргументы не поддерживают ту же семантику, что и аргументы метода. Вы не можете определить аргументы по умолчанию или блокировать аргументы.

Это исправлено только в 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' }
Йорг Миттаг
источник
3
На самом деле, я считаю, что блочные аргументы на define_method поддерживают сплат, который может обеспечить обходной способ определения аргументов по умолчанию.
Чиназавр
1
Чиназавр прав в отношении аргументов блока, допускающих сплат. Я подтвердил это в Ruby 1.8.7 и 1.9.1.
Питер Вагенет
Спасибо, я забыл об этом. Исправлено сейчас.
Йорг Миттаг