Struct允许我创建一个新的类,它接受参数并具有一些很好的语义.但是,参数不是必需的,它们的顺序需要参考定义:
Point = Struct.new(:x,:y) Point.new(111,222) #=> <point instance with x = 111,y = 222> Point.new(111) #=> <point instance with x = 111,y = nil>
我想要类似于Struct的东西,但它使用关键字参数代替:
Point = StricterStruct.new(:x,:y) Point.new(x: 111,y: 222) #=> <point instance with x = 111,y = 222> Point.new(x: 111) #=> ArgumentError
这可能看起来像这样:
module StricterStruct def self.new(*attributes) klass = Class.new klass.instance_eval { ... } klass end end
但是在大括号中应该怎么做才能在klass上定义一个初始化方法:
>它需要没有默认值的关键字参数;
>关键字作为属性中的符号数组给出;和
> initialize方法将它们分配给同名的实例变量
解决方法
由于Ruby 2.0中的新功能,我使用(令人惊讶的Pythonic)** kwargs策略结束了:
module StricterStruct def self.new(*attribute_names_as_symbols) c = Class.new l = attribute_names_as_symbols c.instance_eval { define_method(:initialize) do |**kwargs| unless kwargs.keys.sort == l.sort extra = kwargs.keys - l missing = l - kwargs.keys raise ArgumentError.new <<-MESSAGE keys do not match expected list: -- missing keys: #{missing} -- extra keys: #{extra} MESSAGE end kwargs.map do |k,v| instance_variable_set "@#{k}",v end end l.each do |sym| attr_reader sym end } c end end