define_method 带有预定义的关键字参数
define_method with predefined keyword arguments
我想定义一个采用关键字参数的方法。我希望它在未提供关键字参数时引发,我可以自己编写代码 - 但理想情况下我想让 Ruby 为我做这件事。我也希望能够使用 Method#parameters
检查新定义的方法。如果我使用 shorthand double-splat(如 **kwargs
),我期望的实际结构对 parameters
.
不可见
我当然可以这样做:
define_method(:foo) do | foo:, bar: |
# ...
end
达到预期效果:
method(:foo).parameters
# => [[:keyreq, :foo], [:keyreq, :bar]]
但我不能以编程方式传递这些参数,它们必须按字面意思放在代码中。有什么办法可以绕过这个吗?
您必须使用 eval
来动态定义参数(不仅仅是关键字参数),例如使用 class_eval
:
class MyClass
name = :foo
args = [:bar, :baz]
class_eval <<-METHOD, __FILE__, __LINE__ + 1
def #{name}(#{args.map { |a| "#{a}:" }.join(', ')}) # def foo(bar:, baz:)
[#{args.join(', ')}] # [bar, baz]
end # end
METHOD
end
MyClass.new.foo(bar: 1, baz: 2)
#=> [1, 2]
MyClass.instance_method(:foo).parameters
#=> [[:keyreq, :bar], [:keyreq, :baz]]
我想定义一个采用关键字参数的方法。我希望它在未提供关键字参数时引发,我可以自己编写代码 - 但理想情况下我想让 Ruby 为我做这件事。我也希望能够使用 Method#parameters
检查新定义的方法。如果我使用 shorthand double-splat(如 **kwargs
),我期望的实际结构对 parameters
.
我当然可以这样做:
define_method(:foo) do | foo:, bar: |
# ...
end
达到预期效果:
method(:foo).parameters
# => [[:keyreq, :foo], [:keyreq, :bar]]
但我不能以编程方式传递这些参数,它们必须按字面意思放在代码中。有什么办法可以绕过这个吗?
您必须使用 eval
来动态定义参数(不仅仅是关键字参数),例如使用 class_eval
:
class MyClass
name = :foo
args = [:bar, :baz]
class_eval <<-METHOD, __FILE__, __LINE__ + 1
def #{name}(#{args.map { |a| "#{a}:" }.join(', ')}) # def foo(bar:, baz:)
[#{args.join(', ')}] # [bar, baz]
end # end
METHOD
end
MyClass.new.foo(bar: 1, baz: 2)
#=> [1, 2]
MyClass.instance_method(:foo).parameters
#=> [[:keyreq, :bar], [:keyreq, :baz]]