How do you pass arguments to define_method?
我想将一个参数传递给使用define_method定义的方法,我该怎么做?
传递给define_method的块可以包含一些参数。这就是您定义的方法接受参数的方式。当您定义一个方法时,您实际上只是在给该块起个昵称并在类中保留对其的引用。参数随块一起提供。所以:
1
| define_method(:say_hi) { |other| puts"Hi," + other } |
...以及是否需要可选参数
1 2 3 4 5 6 7 8 9 10 11
| class Bar
define_method(:foo) do |arg=nil|
arg
end
end
a = Bar.new
a.foo
#=> nil
a.foo 1
# => 1 |
...尽可能多的参数
1 2 3 4 5 6 7 8 9 10 11 12 13
| 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'] |
...的组合
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| 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] |
... 他们全部
1 2 3 4 5 6 7 8 9 10 11 12
| 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 introduced keyword arguments, and ** acts like *, but for keyword arguments. It returns a Hash with key / value pairs.
...当然,您也可以在define方法中使用它:)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| 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} |
命名属性示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| 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 2
| define_method(:foo) do |variable1, variable2,*arg, i_will_not: 'work', **options, &block|
# ... |
要么
1 2
| 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
除了Kevin Conner的回答:块参数不支持与方法参数相同的语义。您不能定义默认参数或块参数。
仅在Ruby 1.9中使用新的" stabby lambda"语法来修复此问题,该语法支持完整的方法参数语义。
例:
1 2 3 4 5 6 7 8
| # 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' } |
使用2.2,您现在可以使用关键字参数:
https://robots.thoughtbot.com/ruby-2-keyword-arguments
1 2 3
| define_method(:method) do |refresh: false|
..........
end |
|