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

ruby – 分配分支条件太高

我有一个简单的类,在初始化时,需要一到八个参数.它将访问器设置为稍后使用. Rubocop正试图逮捕我,因为ABC太高了,但我不确定我所做的事情是否真的有问题.这是我刚刚在初始化时禁用检查的情况吗?
class Foo
  attr_accessor :one,:two,:three,:four
  attr_accessor :five,:six,:seven,:eight

  def initialize(p={})
    @one = p[:one] if p[:one].present?
    # ...
    @eight = p[:eight] if p[:eight].present?
  end
end

我对减小大小的唯一想法是做一些事情,比如在初始化时迭代我的所有attr_accessors,看看是否在has中传递了相应的符号,如果是这样的话.

class Foo
  attr_accessor :one,:eight

  def initialize(p={})
    instance_variables.each do |variable|
      send("@#{variable}") = p["#{send(variable)}".to_sym] if p["#{send(variable)}".to_sym].present?
    end
  end
end

但这似乎有点弱.

解决方法

以下是实现您要做的事情的方法之一:
class Foo
  %i(one two three four five six seven eight).each { |attribute| attr_accessor attribute }

  def initialize(p = {})
    p.keys.each { |k| instance_variable_set("@#{k}",p.fetch(k,nil)) }
  end
end

查看Hash#fetch方法.

您还可以使用它来访问p变量的键值对,如果您不是8个变量决定使用一个变量(@p)

编辑

只是出于好奇而写了这个版本(使用了一些元编程) – 它将为添加的实例变量动态添加attr_accessor:

class Foo
  def initialize(p = {})
    p.keys.each do |k|
      instance_variable_set("@#{k}",nil))
      self.class.__send__(:attr_accessor,k)
    end
  end
end

发生了什么,我们提供初始化方法参数(散列p),获取其键并从中创建实例变量,为每个变量分配与键对应的值.然后我们为每个键定义attr_accessor.

a = Foo.new(a: 2,b: 3)
#=> #<Foo:0x00000002d63ad8 @a=2,@b=3>

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

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

相关推荐