当您可以在#describe范围内运行代码时,Mocha的#beforeEach的目的是什么?

时间:2016-10-19 08:25:43

标签: javascript unit-testing testing tdd mocha

我从这个article学习TDD,并且作者讨论了Mocha的beforeEach如何在每个断言之前运行代码。但是,当您只需在描述范围内运行代码时,我就不明白为什么需要这样做。

describe('Test suite for UserComponent', () => {
  beforeEach(() => {
    // Prevent duplication
    wrapper = shallow(<UserComponent
                            name={ 'Reign' }
                            age={ 26 } />);
  });

  it('UserComponent should exist', () => {
    expect(wrapper).to.exist;
  });

  it('Correctly displays the user name and age in paragraphs wrapped under a parent div', () => {
    expect(wrapper.type()).to.equal('div');
    // more code...
  });
});

但是不使用beforeEach仍然有效 -

describe('Test suite for UserComponent', () => {

wrapper = shallow(<UserComponent
                        name={ 'Reign' }
                        age={ 26 } />);

  it('UserComponent should exist', () => {
    expect(wrapper).to.exist;
  });

  it('Correctly displays the user name and age in paragraphs wrapped under a parent div', () => {
    expect(wrapper.type()).to.equal('div');
    // more code...
  });
});

1 个答案:

答案 0 :(得分:2)

每次测试之前执行

beforeEach。当代码从beforeEach移动到传递给describe的函数内部时,此迭代将丢失。但是,这还不是全部。

某些情况下,直接在传递给describe 的函数内执行的代码可以执行与beforeEach挂钩相同的任务。例如,如果它的函数是初始化describe块中测试的本地只读结构,那么你可以跳过beforeEach挂钩。

然而,Mocha立即执行传递给describe个调用的所有回调,而beforeEach调用注册传递给它的函数以供将来执行,它将会只在需要时执行 。如果初始化很昂贵,最好使用beforeEach钩子,因为如果你使用--grep来只选择一些测试,或者使用it.only来运行单个测试,那么Mocha会只有当它与实际运行的测试相关时才运行钩子。 如果您在describe中有初始化代码,则Mocha无法跳过它,因此您每次都需要支付初始化费用但是,如果您打算使用一个钩子,数据是不可变的,然后before优于beforeEach,因为它只运行一次而不是每次测试之前。

然后有些情况下直接在传递给describe的函数中运行代码根本不起作用。想象一下,sharedResource是一个资源,所有测试需要使用。例如,它可能是一个携带国家的第三方图书馆。有些测试需要将其设置为某种状态。其他测试需要它处于不同的状态。这不起作用:

"use strict";

const assert = require('assert');

let sharedResource;

describe("in condition a", () => {
    sharedResource = true;

    it("sharedResource is true", () => assert(sharedResource));
});

describe("in condition b", () => {
    sharedResource = false;

    it("sharedResource is false", () => assert(!sharedResource));
});

第一个测试将失败,因为关键语句的执行顺序是:

  1. sharedResource = true;
  2. sharedResource = false;
  3. assert(sharedResource);
  4. assert(!sharedResource);
  5. 使用beforeEach可以轻松解决问题。这很好:

    "use strict";
    
    const assert = require('assert');
    
    let sharedResource;
    
    describe("in condition a", () => {
        beforeEach(() => {
            sharedResource = true;
        });
    
        it("sharedResource is true", () => assert(sharedResource));
    });
    
    describe("in condition b", () => {
        beforeEach(() => {
            sharedResource = false;
        });
    
        it("sharedResource is false", () => assert(!sharedResource));
    });