Як ви передаєте аргументи на define_method?


155

Я хотів би передати аргументи (аргументи) методу, визначеному за допомогою define_method, як би це зробити?

Відповіді:


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 представив аргументи ключових слів, і ** діє як *, але для аргументів ключових слів. Він повертає Hash з парами ключ / значення.

... і звичайно, ви можете використовувати його і в методі defini :)

 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 "захоплює всі решта аргументів", а подвійний splat - "захоплення всіх решти аргументів ключового слова", тому змішування їх порушить очікувану логіку. (Я не маю жодної посилання, щоб довести цю точку доху!)

оновлення 2018 серпня:

Підсумкова стаття: https://blog.eq8.eu/til/metaprogramming-ruby-examples.html


Цікаво - спеціально 4-й блок: він працював на 1.8.7! Перший блок не працював у 1.8.7, а другий блок має помилку друку (має бути a.foo 1замість foo 1). Дякую!
Sony Santos

1
спасибі за відгук, помилка була виправлена, ... На рубінах 1.9.3 та 1.9.2 всі приклади працює, і я впевнений, що на 1.9.1 теж (але не намагався)
еквівалент8

Я поєднав цю відповідь із прийнятою відповіддю на stackoverflow.com/questions/4470108/…, щоб зрозуміти, як перезаписати (не переосмислити) метод під час виконання, який приймає необов’язкові аргументи та блок та все ще зможе викликати оригінальний метод із арґугами і блокувати. Ах, рубін. Зокрема, мені потрібно було замінити Savon :: Client.request у моїй програмі розроблення для одного виклику 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 роблять підтримку splat, яка також може забезпечувати спосіб визначення аргументів за замовчуванням.
Chinasaur

1
Chinasaur вірно відноситься до блокових аргументів, що дозволяють робити заставки. Я підтвердив це і в Ruby 1.8.7 та 1.9.1.
Пітер Вагенет

Спасибі, я забув про це. Виправлено зараз.
Йорг W Міттаг

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.