变量范围在 Mocha 测试框架中是如何工作的?

Posted

技术标签:

【中文标题】变量范围在 Mocha 测试框架中是如何工作的?【英文标题】:How does variable scope work within the Mocha test framework? 【发布时间】:2015-06-09 00:08:20 【问题描述】:

我是 javascript、node.js、mocha 等所有东西的相对新手。

在我的代码中,我有一个 Unit 对象,它有一个将 disabled 属性设置为 true 的 disable() 和一个返回 disabled 属性的 isDisabled()。它还有一个方法nextTurnReset(),可以在下一回合开始时重置单位。我已经编写了一个测试套件来测试这种行为。我首先禁用该对象,然后尝试测试它是否被禁用。但是,我的第一个测试中的单元变量(在传递给 Mocha 的 it() 方法的匿名函数中)处于非禁用状态,正如我在节点的调试器中观察到的那样。

describe('#disable()', function() 
    var unit = tests.newUnit();
    unit.disable();
    debugger;
    it('disabled off turn?', function() 
        debugger;
        (unit.isDisabled()).should.be.exactly(true);
    );
    unit.nextTurnReset();
    it('disabled on next turn?', function() 
        (unit.isDisabled()).should.be.exactly(true);
    );
    unit.nextTurnReset();
    it('disabled on 2nd turn?', function() 
        (unit.isDisabled()).should.be.exactly(false);
    );
);

作为记录,前两个测试失败,最后一个成功,表明该单元从未被禁用。

使用节点调试器的repl:在第一个debugger; 语句之后,unit.disabled == true,但在第二个debugger; 语句unit.disabled == false 之后。我希望这两种情况下的价值都是真实的。

知道为什么会这样吗?另外,编写 Mocha 测试以获得预期结果的正确方法是什么?

非常感谢!

【问题讨论】:

【参考方案1】:

Mocha 中的变量作用域与任何其他 JavaScript 代码完全相同。您的问题是您没有意识到 Mocha 将执行您的代码的顺序。这就是发生的事情:

    您创建您的单元实例并在其上调用disable

    您注册了您的第一个测试。这就是it 所做的:它为future 执行注册一个测试。测试现在不执行

    您调用 unit.nextTurnReset(); 会重置对象的状态。

    您注册了第二个测试。 它现在不执行

    您再次重置对象。

    您注册了最后一次测试。

之后,Mocha 会接受您注册的测试并运行它们。 在您的测试运行时,您的对象处于重置状态,而不是禁用。

在我看来,鉴于您描述的期望行为,您的代码应该是:

describe('#disable()', function() 
    var unit = tests.newUnit();

    beforeEach(function () 
        unit.nextTurnReset();
    );

    it('disabled off turn?', function() 
        unit.disable();
        (unit.isDisabled()).should.be.exactly(true);
    );

    it('disabled on next turn?', function() 
        (unit.isDisabled()).should.be.exactly(false);
    );
);

传递给beforeEach 的代码在您注册的每个测试之前运行,因此它会在正确的时间重置对象。

【讨论】:

【参考方案2】:

在示例之间传递状态不是一个很好的做法。如果您需要以随机顺序运行测试会发生什么?或者项目中的某个人决定移动示例?

对我来说,拥有以下两个示例似乎足以正确测试 Unit#disable

describe('#disable()', function() 
    it('gets disabled when called on an enabled', function() 
        var unit = tests.newUnit();
        unit.disable();

        (unit.isDisabled()).should.be.exactly(true);
    );

    it('gets enabled when called on a disabled', function() 
        var unit = tests.newUnit();
        unit.disable();
        unit.disable();

        (unit.isDisabled()).should.be.exactly(false);
    );
);

【讨论】:

OP 做错了,但在测试之间将夹具重置为已知状态是一种完全有效的策略。如果整个夹具的创建不繁琐,我更喜欢在每次测试中重新创建它,但有时最好在测试之间重置为已知状态,而不是重新创建昂贵的夹具。在beforeEach 挂钩中调用unit.nextTurnReset() 会将测试夹具重置为 测试之间的已知状态。由于它在beforeEach 挂钩中,因此在每次测试之前对unit.nextTurnReset() 的调用不依赖于哪些测试运行或不运行,或测试顺序。

以上是关于变量范围在 Mocha 测试框架中是如何工作的?的主要内容,如果未能解决你的问题,请参考以下文章

Karma 与测试框架 Jasmine、Mocha、QUnit [关闭]

node----mocha单元测试框架-----格尔尼卡ぃ

前端单元测试框架-Mocha

当我循环测试用例时,为啥 Mocha 会跳过重新分配变量?

Mocha 单元测试框架简介

Spectron,mocha和chai可以声称变量在Electron-app中具有预期值吗?