Как передать аргументы в define_method?
-
01-07-2019 - |
Вопрос
Я хотел бы передать аргумент(ы) методу, определяемому с помощью define_method, как мне это сделать?
Решение
Блок, который вы передаете в 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 появились аргументы ключевых слов, а ** действует так же, как *, но для аргументов ключевых слов.Он возвращает хэш с парами ключ/значение.
... и, конечно, вы можете использовать его и в методе определения :)
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 «захватывает все оставшиеся аргументы», а двойной splat «захватывает все оставшиеся аргументы ключевого слова», поэтому их смешивание нарушит ожидаемую логику.(У меня нет никаких ссылок, подтверждающих это, хах!)
обновление 2018 август:
Итоговая статья: https://blog.eq8.eu/til/metaprogramming-ruby-examples.html
В дополнение к ответу Кевина Коннера:Аргументы блока не поддерживают ту же семантику, что и аргументы метода.Вы не можете определять аргументы по умолчанию или блокировать аргументы.
Это исправлено только в Ruby 1.9 с помощью нового альтернативного синтаксиса «stabby лямбда», который поддерживает полную семантику аргументов метода.
Пример:
# 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