Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/csharp/325.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Ruby 如何传递参数来定义方法?_Ruby_Metaprogramming - Fatal编程技术网

Ruby 如何传递参数来定义方法?

Ruby 如何传递参数来定义方法?,ruby,metaprogramming,Ruby,Metaprogramming,我想将一个或多个参数传递给使用define\u method定义的方法,我该如何做?传递给define\u method的块可以包含一些参数。这就是您定义的方法接受参数的方式。当你定义一个方法的时候,你实际上只是在给这个块做小标记,并在类中保留对它的引用。参数随块一起提供。因此: define_method(:say_hi) { |other| puts "Hi, " + other } 除了Kevin Conner的回答之外:块参数不支持与方法参数相同的语义。不能定义默认参数或块参数 这仅在

我想将一个或多个参数传递给使用define\u method定义的方法,我该如何做?

传递给define\u method的块可以包含一些参数。这就是您定义的方法接受参数的方式。当你定义一个方法的时候,你实际上只是在给这个块做小标记,并在类中保留对它的引用。参数随块一起提供。因此:

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

除了Kevin Conner的回答之外:块参数不支持与方法参数相同的语义。不能定义默认参数或块参数

这仅在Ruby1.9中通过支持完整方法参数语义的新的替代“ThankyLambda”语法修复

例如:

# 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' }

。。。如果你想要可选的参数

 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引入了双splat
**
(双星),它()可以:

Ruby 2.0引入了关键字参数,**的行为类似于*,但用于关键字参数。它返回一个包含键/值对的散列

…当然,您也可以在define方法中使用它:)

命名属性示例:

 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}
我试图创建一个包含关键字argument、splat和double splat的示例:

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

。。。但这是行不通的,看起来有一个限制。当您考虑它是有意义的,因为splat操作符是“捕获所有剩余的参数”,而double splat是“捕获所有剩余的关键字参数”,因此混合它们会破坏预期的逻辑。(我没有任何参考资料来证明这一点

2018年8月更新:


摘要文章:

使用2.2,您现在可以使用关键字参数:


实际上,我相信define_方法上的块参数确实支持splat,它也可以提供一种全面定义默认参数的方法。Chinasaur关于允许splat的块参数是正确的。我在Ruby 1.8.7和1.9.1中都证实了这一点。谢谢,我忘了这一点。现在修好了。很有意思-特别是第四个区块:它在1.8.7上确实有效!第一个块在1.8.7中不起作用,第二个块有输入错误(应该是
a.foo 1
,而不是
foo 1
)。谢谢你!感谢您的反馈,拼写错误已经修复,…在ruby 1.9.3和1.9.2上,所有示例都有效,我确信在1.9.1上(但没有尝试)我也将此答案与在上接受的答案结合起来,以了解如何覆盖(而不是覆盖)一种在运行时接受可选参数和块的方法,它仍然能够使用参数和块调用原始方法。啊,鲁比。具体来说,我需要在我的dev env中覆盖Savon::Client.request,以便对只能在生产环境中访问的主机进行单个API调用。干杯好吧,那只是一件纯粹的美好的事情。干得好,凯文·科斯特纳。
 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|
    # ...
define_method(:method) do |refresh: false|
  ..........
end