你见过这样声明的函数吗?

def foo a, **b
  ...
end

我知道单个*是splat操作符。**是什么意思?


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

对于这段代码:

def foo(a, *b, **c)
  [a, b, c]
end

下面是一个演示:

> foo 10
=> [10, [], {}]
> foo 10, 20, 30
=> [10, [20, 30], {}]
> foo 10, 20, 30, d: 40, e: 50
=> [10, [20, 30], {:d=>40, :e=>50}]
> foo 10, d: 40, e: 50
=> [10, [], {:d=>40, :e=>50}]

这是自Ruby 2.0以来可用的double splat运算符。

它捕获所有关键字参数(也可以是简单的散列,这是在关键字参数成为Ruby语言的一部分之前模拟关键字参数的惯用方法)

def my_method(**options)
  puts options.inspect
end

my_method(key: "value")

上面的代码将{key:value}输出到控制台。

就像单个splat操作符捕获所有常规参数一样,但您得到的不是数组,而是散列。

现实生活中的例子:

例如,在Rails中,循环方法是这样的:

def cycle(first_value, *values)
  options = values.extract_options!
  # ...
end

这个方法可以这样调用:cycle(“red”,“green”,“blue”,名称:“colors”)。

这是一个非常常见的模式:您接受一个参数列表,最后一个是一个选项散列,它可以被提取—例如—使用ActiveSupport的extract_options!

在Ruby 2.0中,你可以简化这些方法:

def cycle(first_value, *values, **options)
  # Same code as above without further changes!
end

诚然,如果您已经在使用ActiveSupport,这只是一个小的改进,但对于纯Ruby,代码获得了相当多的简练。


此外,你可以像这样在调用方使用它:

def foo(opts); p opts end
bar = {a:1, b:2}

foo(bar, c: 3)
=> ArgumentError: wrong number of arguments (given 2, expected 1)

foo(**bar, c: 3)
=> {:a=>1, :b=>2, :c=>3}

就你的情况而言,以上答案是准确的。

然而,双splat运算符也可以在Ruby语言中用于算术运算。例如,x^y可以写成x**y。

更多信息请参考这个stackover问题。