Rspec:期望与期望与阻止 - 差异是什么?

时间:2013-11-13 17:48:56

标签: ruby tdd rspec2

刚学习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中有什么区别?
  3. 我也在寻找规则,关于什么时候使用其他

    相同但反向结果的另一个例子,这段代码可以工作:

      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
    

2 个答案:

答案 0 :(得分:48)

如前所述:

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方法执行任何块的方法被传递给方法。

答案 1 :(得分:9)

在第一种情况下,当您将块传递给expect时,在评估结果之前不会执行块的执行,此时RSpec代码可以捕获引发的任何错误并按照预期进行检查。

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

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