define_methodに引数をどのように渡しますか?


155

define_methodを使用して定義されているメソッドに引数を渡したいのですが、どうすればよいですか?

回答:


198

define_methodに渡すブロックには、いくつかのパラメーターを含めることができます。これが、定義したメソッドが引数を受け入れる方法です。メソッドを定義するときは、ブロックにニックネームを付けて、クラスへの参照を維持するだけです。パラメータはブロックに付属しています。そう:

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

まあそれは純粋で純粋なビーティのことです。よくできました、ケビンコスナー。
Darth Egregious

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は、二重引用符**(2つの星)を導入しました(引用します)。

Ruby 2.0ではキーワード引数が導入されており、**は*と同じように機能しますが、キーワード引数に対してです。キーと値のペアを含むハッシュを返します。

...もちろん、defineメソッドでも使用できます:)

 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を1つにまとめた例を作成しようとしていました。

 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演算子は「残りのすべての引数を取り込む」ことであり、double splatは「残りのすべてのキーワード引数を取り込む」ことであるため、それらを混在させると予想されるロジックが壊れてしまいます。(この点を証明するための参照はありません!)

2018年8月の更新:

概要記事:https : //blog.eq8.eu/til/metaprogramming-ruby-examples.html


興味深い-特に4番目のブロック:1.8.7で動作しました!1.8.7では最初のブロックが機能せず、2番目のブロックにタイプミスがあります(のa.foo 1代わりにすべきですfoo 1)。ありがとう!
ソニーサントス

1
フィードバックのおかげで、誤植が修正されました... ... ruby​​ 1.9.3と1.9.2ではすべての例が機能し、1.9.1でも動作することを確信しています(ただし試していませんでした)
同等8

この回答とstackoverflow.com/questions/4470108/…で承認された回答を組み合わせて、実行時にオプションの引数とブロックを受け取るメソッドを上書きする(上書きしない)方法と、引数を使用して元のメソッドを呼び出す方法を理解するとブロック。あ、ルビー。具体的には、本番環境でのみアクセスできるホストへの単一のAPI呼び出しのために、開発環境のSavon :: Client.requestを上書きする必要がありました。乾杯!
pduey

59

Kevin Connerの回答に加えて、ブロック引数はメソッド引数と同じセマンティクスをサポートしていません。デフォルト引数またはブロック引数を定義することはできません。

これは、完全なメソッド引数のセマンティクスをサポートする新しい代替「stabby lambda」構文を備えたRuby 1.9でのみ修正されています。

例:

# 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をサポートしていると思います。これは、デフォルトの引数を定義するための大まかな方法​​を提供できます。
2009

1
Chinasaurは、スプラットを許可するブロック引数については正しいです。Ruby 1.8.7と1.9.1の両方でこれを確認しました。
Peter Wagenet、2009年

ありがとう、これを忘れてしまいました。今修正されました。
イェルクWミッターク

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.