Відповіді:
Блок, який ви переходите до define_method, може включати деякі параметри. Ось як ваш визначений метод приймає аргументи. Коли ви визначаєте метод, ви насправді просто називаєте блок і зберігаєте посилання на нього у класі. Параметри поставляються з блоком. Так:
define_method(:say_hi) { |other| puts "Hi, " + other }
... і якщо потрібно необов'язкові параметри
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
a.foo 1
замість foo 1
). Дякую!
Окрім відповіді Кевіна Коннера: аргументи блоків не підтримують ту саму семантику, як аргументи методу. Ви не можете визначити аргументи за замовчуванням або блокувати аргументи.
Це зафіксовано лише в 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' }
З 2.2 тепер ви можете використовувати аргументи ключових слів: https://robots.thoughtbot.com/ruby-2-keyword-arguments
define_method(:method) do |refresh: false|
..........
end