Come si passano gli argomenti a define_method?


155

Vorrei passare un argomento (s) a un metodo che viene definito usando define_method, come lo farei?

Risposte:


198

Il blocco che si passa a define_method può includere alcuni parametri. Ecco come il tuo metodo definito accetta argomenti. Quando definisci un metodo, stai semplicemente soprannominando il blocco e mantenendo un riferimento ad esso nella classe. I parametri vengono forniti con il blocco. Così:

define_method(:say_hi) { |other| puts "Hi, " + other }

Bene, questa è solo una questione di pura bellezza pura. Bel lavoro, Kevin Costner.
Darth Egregious,

90

... e se vuoi parametri opzionali

 class Bar
   define_method(:foo) do |arg=nil|                  
     arg                                                                                          
   end   
 end

 a = Bar.new
 a.foo
 #=> nil
 a.foo 1
 # => 1

... quanti argomenti vuoi

 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']

...combinazione di

 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]

... tutti loro

 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

Aggiornare

Ruby 2.0 ha introdotto il doppio splat **(due stelle) che ( cito ) fa:

Ruby 2.0 ha introdotto argomenti di parole chiave e ** si comporta come *, ma per argomenti di parole chiave. Restituisce un hash con coppie chiave / valore.

... e ovviamente puoi usarlo anche nel metodo di definizione :)

 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}

Esempio di attributi nominati:

 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}

Stavo cercando di creare un esempio con l'argomento keyword, splat e double splat tutto in uno:

 define_method(:foo) do |variable1, variable2,*arg, i_will_not: 'work', **options, &block|
    # ...

o

 define_method(:foo) do |variable1, variable2, i_will_not: 'work', *arg, **options, &block|
    # ...

... ma questo non funzionerà, sembra che ci sia un limite. Se ci pensate, ha senso in quanto l'operatore splat sta "catturando tutti gli argomenti rimanenti" e double splat sta "catturando tutti gli argomenti rimanenti delle parole chiave", quindi mescolarli romperebbe la logica prevista. (Non ho alcun riferimento per dimostrare questo punto doh!)

aggiornamento agosto 2018:

Articolo di sintesi: https://blog.eq8.eu/til/metaprogramming-ruby-examples.html


Interessante - in particolare il 4 ° blocco: ha funzionato su 1.8.7! Il primo blocco non ha funzionato in 1.8.7 e il secondo blocco ha un refuso (dovrebbe essere a.foo 1invece di foo 1). Grazie!
Sony Santos,

1
grazie per il feedback, l'errore di battitura è stato corretto, ... Su ruby ​​1.9.3 e 1.9.2 tutti gli esempi funzionano e sono sicuro che anche su 1.9.1 (ma non ci ho provato)
equivalente 8

Ho combinato questa risposta con la risposta accettata su stackoverflow.com/questions/4470108/… per capire come sovrascrivere (non sovrascrivere) un metodo in fase di runtime che accetta args opzionali e un blocco ed essere ancora in grado di chiamare il metodo originale con args e bloccare. Ah, rubino. In particolare, dovevo sovrascrivere Savon :: Client.request nel mio sviluppatore per una singola chiamata API a un host a cui posso accedere solo in produzione. Saluti!
pduey,

59

Oltre alla risposta di Kevin Conner: gli argomenti di blocco non supportano la stessa semantica degli argomenti di metodo. Non è possibile definire argomenti predefiniti o argomenti a blocchi.

Ciò è stato risolto solo in Ruby 1.9 con la nuova sintassi alternativa "stabby lambda" che supporta la semantica dell'argomento metodo completo.

Esempio:

# 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
In realtà, credo che gli argomenti a blocchi su define_method supportino splat, che può fornire un modo round-a-bout per definire anche gli argomenti predefiniti.
Chinasaur,

1
Chinasaur ha ragione riguardo agli argomenti a blocchi che consentono gli splat. L'ho confermato in Ruby 1.8.7 e 1.9.1.
Peter Wagenet,

Grazie, me ne sono dimenticato. Riparato ora.
Jörg W Mittag,

Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.