Como você passar argumentos para define_method?
-
01-07-2019 - |
Pergunta
Gostaria de passar um argumento (s) a um método que está sendo definida usando define_method, como eu faria isso?
Solução
O bloco que você passa para define_method pode incluir alguns parâmetros. Isso é como o seu método definido aceita argumentos. Quando você define um método que você está realmente apenas apelidando o bloco e manter uma referência a ele na classe. Os parâmetros vêm com o bloco. Assim:
define_method(:say_hi) { |other| puts "Hi, " + other }
Outras dicas
... e se você quiser parâmetros opcionais
class Bar
define_method(:foo) do |arg=nil|
arg
end
end
a = Bar.new
a.foo
#=> nil
a.foo 1
# => 1
... tantos argumentos como você deseja
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']
... combinação de
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]
... todos eles
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
Atualizar
Rubi 2.0 introduziu **
splat duplo (duas estrelas), que ( cito ) faz:
Rubi 2.0 introduziu argumentos chave, e ** age como *, mas para argumentos nomeados. Ele retorna um Hash com pares de chave / valor.
... e, claro, você pode usá-lo em definir método também :)
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}
atributos nomeados exemplo:
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}
Eu estava tentando criar exemplo, com argumento de palavra-chave, splat e splat double tudo em um:
define_method(:foo) do |variable1, variable2,*arg, i_will_not: 'work', **options, &block|
# ...
ou
define_method(:foo) do |variable1, variable2, i_will_not: 'work', *arg, **options, &block|
# ...
... mas isso não vai funcionar, parece que há uma limitação. Quando você pensa sobre isso faz sentido como operador de splat é "capturar todos os argumentos restantes" e duplo splat é "capturar todos os restantes argumentos de palavra-chave", portanto, misturá-los iria quebrar a lógica esperado. (Eu não tenho qualquer referência para provar isso doh ponto!)
update 2018 Agosto:
Resumo do artigo: https://blog.eq8.eu/til/metaprogramming -ruby-examples.html
Além de resposta de Kevin Conner: argumentos bloco não suportam a mesma semântica como argumentos de método. Você não pode definir argumentos padrão ou argumentos bloco.
Isso só é fixado no Ruby 1.9 com a nova alternativa sintaxe "lambda stabby" que suporta método integral semântica argumento.
Exemplo:
# 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' }
Com 2.2 agora você pode usar argumentos de palavra-chave: https://robots.thoughtbot.com/ruby-2-keyword-arguments
define_method(:method) do |refresh: false|
..........
end