微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

以元编程方式定义采用关键字参数的Ruby方法?

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

原文地址:https://www.jb51.cc/ruby/269700.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐