变量范围在 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 测试框架中是如何工作的?的主要内容,如果未能解决你的问题,请参考以下文章