39

刚刚学习 rspec 语法,我注意到这段代码有效:

  context "given a bad list of players" do
    let(:bad_players) { {} }

    it "fails to create given a bad player list" do
       expect{ Team.new("Random", bad_players) }.to raise_error
     end 
  end

但是这段代码没有:

  context "given a bad list of players" do
    let(:bad_players) { {} }

    it "fails to create given a bad player list" do
       expect( Team.new("Random", bad_players) ).to raise_error
     end 
  end

它给了我这个错误:

Team given a bad list of players fails to create given a bad player list
     Failure/Error: expect( Team.new("Random", bad_players) ).to raise_error
     Exception:
       Exception
     # ./lib/team.rb:6:in `initialize'
     # ./spec/team_spec.rb:23:in `new'
     # ./spec/team_spec.rb:23:in `block (3 levels) in <top (required)>'

我的问题是:

  1. 为什么会这样?
  2. 在 ruby​​ 中,前一个示例和后一个示例有什么区别?

我也在寻找关于何时使用一个而不是另一个的规则

另一个相同但相反结果的示例,其中此代码有效:

  it "has a list of players" do
    expect(Team.new("Random").players).to be_kind_of Array
  end 

但是这段代码失败了

  it "has a list of players" do
    expect{ Team.new("Random").players }.to be_kind_of Array
  end

在这种情况下我得到的错误是:

Failure/Error: expect{ Team.new("Random").players }.to be_kind_of Array
       expected #<Proc:0x007fbbbab29580@/Users/amiterandole/Documents/current/ruby_sandbox/tdd-ruby/spec/team_spec.rb:9> to be a kind of Array
     # ./spec/team_spec.rb:9:in `block (2 levels) in <top (required)>'

我正在测试的课程如下所示:

class Team
  attr_reader :name, :players

  def initialize(name, players = [])
    raise Exception unless players.is_a? Array

    @name = name
    @players = players
  end
end
4

2 回答 2

70

如前所述:

expect(4).to eq(4)

这是专门测试您作为参数发送到方法的值。当您尝试在执行相同操作时测试引发的错误时:

expect(raise "fail!").to raise_error

您的论点会立即被评估,并且该异常将被抛出,您的测试将在那里爆炸。

但是,当您使用块时(这是基本的 ruby​​),块内容不会立即执行 - 它的执行取决于您正在调用的方法(在这种情况下,该expect方法处理何时执行您的块):

expect{raise "fail!"}.to raise_error

我们可以看一个可能处理这种行为的示例方法:

def expect(val=nil)
  if block_given?
    begin
      yield
    rescue
      puts "Your block raised an error!"
    end
  else
    puts "The value under test is #{val}"
  end
end

您可以在此处看到,它expect是手动拯救错误的方法,以便它可以测试是否引发错误等。 yield这是 ruby​​ 方法执行传递给该方法的任何块的方法。

于 2013-11-13T19:09:36.670 回答
13

在第一种情况下,当您将一个块传递给 时expect,直到评估结果时才会执行该块,此时 RSpec 代码可以捕获引发的任何错误并根据预期检查它。

在第二种情况下,在expect评估参数时会引发错误,因此expect代码没有机会参与其中。

至于规则,Proc如果您尝试测试行为(例如引发错误,更改某些值),则传递一个块或 a。否则,您传递一个“常规”参数,在这种情况下,该参数的值就是测试的值。

于 2013-11-13T17:59:23.170 回答