使用块初始化一个对象

16

可以使用以下方式用块初始化一个对象吗?

class Foo
  attr_reader :bar,:baz
  def initialize(bar,baz)
    @bar, @baz = bar, baz
  end
end

Foo.new do |bar, baz|
  # some logic to be implemented
  # @bar, @baz to be assigned
end

1
当然,但不要忘记提供参数。而且该块不会执行任何操作。 - sawa
2个回答

29

当然,你可以在initialize中使用yield,没有什么特别的地方:

class Foo
  attr_accessor :bar, :baz
  def initialize
    yield self
  end
end

Foo.new do |f|
  f.bar = 123
  f.baz = 456
end
#=> <Foo:0x007fed8287b3c0 @bar=123, @baz=456>

您还可以使用instance_eval在接收器的上下文中评估该块:

class Foo
  attr_accessor :bar, :baz
  def initialize(&block)
    instance_eval(&block)
  end
end

Foo.new do
  @bar = 123
  @baz = 456
end
#=> #<Foo:0x007fdd0b1ef4c0 @bar=123, @baz=456>

1
如果我想要一个对象可以通过块和构造函数两种方式实例化,该怎么办? - DreamWalker
1
@DreamWalker,你可以将参数设为可选,并且只有在block_given?返回true时才执行yield。 - Stefan

8
如果构造函数的参数是可选的,而且相同的属性是可写的,你可以使用tap来实现类似的效果。在需要设置未由构造函数设置的实例值时,tap特别有用。
class Foo
  attr_accessor :bar,:baz
  def initialize(bar: nil, baz: nil)
    @bar, @baz = bar, baz
  end
end

foo1 = Foo.new(bar: 'bar', baz: 'baz')
foo2 = Foo.new.tap do |f|
  f.bar = 'bar'
  f.baz = 'baz'
end

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接