RSpec - 测试替身


在本章中,我们将讨论 RSpec Doubles,也称为 RSpec Mocks。Double 是一个可以“替代”另一个对象的对象。您可能想知道这到底意味着什么以及为什么需要它。

假设您正在为一所学校构建一个应用程序,并且您有一个代表学生教室的类和另一个代表学生的类,即您有一个 Classroom 类和一个 Student 类。您需要首先为其中一个类编写代码,所以我们从 Classroom 类开始 -

class ClassRoom 
   def initialize(students) 
      @students = students 
   end 
   
   def list_student_names 
      @students.map(&:name).join(',') 
   end 
end

这是一个简单的类,它有一个方法 list_student_names,该方法返回以逗号分隔的学生姓名字符串。现在,我们想为这个类创建测试,但是如果我们还没有创建 Student 类,我们该怎么做呢?我们需要一个测试替身。

另外,如果我们有一个Behave类似于 Student 对象的“虚拟”类,那么我们的 ClassRoom 测试将不依赖于 Student 类。我们称之为测试隔离。

如果我们的 ClassRoom 测试不依赖于任何其他类,那么当测试失败时,我们可以立即知道我们的 ClassRoom 类中存在错误,而不是其他类中存在错误。请记住,在现实世界中,您可能正在构建一个需要与其他人编写的另一个类进行交互的类。

这就是 RSpec Doubles(模拟)发挥作用的地方。我们的 list_student_names 方法调用其 @students 成员变量中每个 Student 对象的 name 方法。因此,我们需要一个实现 name 方法的 Double。

以下是 ClassRoom 的代码以及 RSpec 示例(测试),但请注意,没有定义 Student 类 -

class ClassRoom 
   def initialize(students) 
      @students = students 
   end
   
   def list_student_names 
      @students.map(&:name).join(',') 
   end 
end

describe ClassRoom do 
   it 'the list_student_names method should work correctly' do 
      student1 = double('student') 
      student2 = double('student') 
      
      allow(student1).to receive(:name) { 'John Smith'} 
      allow(student2).to receive(:name) { 'Jill Smith'} 
      
      cr = ClassRoom.new [student1,student2]
      expect(cr.list_student_names).to eq('John Smith,Jill Smith') 
   end 
end

当执行上述代码时,将产生以下输出。您的计算机上经过的时间可能略有不同 -

. 
Finished in 0.01 seconds (files took 0.11201 seconds to load) 
1 example, 0 failures

正如您所看到的,使用测试替身可以让您测试代码,即使它依赖于未定义或不可用的类。此外,这意味着当测试失败时,您可以立即看出这是因为您的类中存在问题,而不是其他人编写的类。