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

최신 정보

루비 2.0 더블 플랫 소개 **((별 2 개) I 견적이 ) 수행을 :

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}

키워드 인수, 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 연산자가 "남은 모든 인수를 캡처"하고 double splat이 "남은 모든 키워드 인수를 캡처"하므로이 논리를 혼합하면 예상되는 논리가 손상 될 수 있습니다. (이 시점을 증명할 어떤 언급도 없습니다!)

2018 년 8 월 업데이트 :

요약 기사 : https://blog.eq8.eu/til/metaprogramming-ruby-examples.html


흥미롭게도-4 번째 블록 : 1.8.7에서 작동했습니다! 첫 번째 블록은 1.8.7에서 작동하지 않았으며 두 번째 블록에는 오타가 있습니다 ( a.foo 1대신 대신 해야 함 foo 1). 감사!
Sony Santos

1
의견을 주셔서 감사합니다, 오타가 루비 1.9.3 및 예제 작품의 1.9.2 모두에 ... 고정하고 있었고, 난 너무 1.9.1에 (하지만 시도하지 않은) 긍정적 해요
equivalent8

나는이 답변을 stackoverflow.com/questions/4470108/ 에서 허용되는 답변과 결합 하여 런타임에서 선택적 args 및 블록을 사용하고 args를 사용하여 원래 메소드를 호출 할 수있는 메소드를 덮어 쓰는 방법 (재정의하지 않음)을 파악했습니다. 차단. 아, 루비 특히 프로덕션 환경에서만 액세스 할 수있는 호스트에 대한 단일 API 호출을 위해 dev env에서 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을 지원하므로 기본 인수를 정의하는 둥근 방법을 제공 할 수 있다고 생각합니다.
Chinasaur 2009

1
Chinasaur는 표시를 허용하는 블록 인수에 대해 정확합니다. Ruby 1.8.7과 1.9.1에서 이것을 확인했습니다.
Peter Wagenet

고마워, 나는 이것에 대해 잊었다. 지금 수정했습니다.
Jörg W Mittag

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.