如何验证在 Moq 中没有调用该方法?

Posted

技术标签:

【中文标题】如何验证在 Moq 中没有调用该方法?【英文标题】:How to verify that method was NOT called in Moq? 【发布时间】:2010-10-06 22:37:59 【问题描述】:

如何验证Moq 中没有调用该方法?

它有类似 AssertWasNotCalled 的东西吗?

更新:从 3.0 版开始,可以使用新语法:

mock.Verify(foo => foo.Execute("ping"), Times.Never());

【问题讨论】:

【参考方案1】:

使用 .AtMostOnce();

真正测试后,再次调用该方法。如果它抛出一个异常,它就会被调用。

【讨论】:

断言异常是由模拟框架抛出的,是不是有点太晦涩难懂了? 为什么?只需检查异常的类型。如果它是我最小起订量的一个扔掉,你是安全的。 对 Times.Never 使用验证是一个更好的选择......我同意 alex 的观点,即这个解决方案有效,但绝对是晦涩难懂的。【参考方案2】:

更新:从第 3 版开始,请查看上述问题的更新或以下 Dann 的回答。

或者,让你的模拟严格,这样如果你调用一个你没有期望的方法,它就会失败

new Mock<IMoq>(MockBehavior.Strict)

或者,如果您希望您的模拟是松散的,请使用 .Throws( Exception )

var m = new Mock<IMoq>(MockBehavior.Loose);
m.Expect(a => a.moo()).Throws(new Exception("Shouldn't be called."));

【讨论】:

... 或 Callback() 设置一些可以断言的标志。 同样使用选项#2,您不能在一般的 Teardown 方法中使用 VerifyAll - 它会失败,说没有达到预期;理想情况下测试应该通过的时间。 这并不是真正的“未调用验证”,因为它可能会在方法中被捕获并且仍然有效 - 提供误报! Expect 现已弃用 这可能是 2009 年最好的方法,但现在肯定不是。对不起【参考方案3】:

这在最新版本的 Moq 中不起作用(至少从 3.1 开始),应在 Verify 方法中指定,如 回答。

其实最好在 Returns 语句后指定.AtMost(0)

var m = new Mock<ISomething>();
m.Expect(x => x.Forbidden()).Returns("foo").AtMost(0);

虽然“投掷”也有效,但AtMost(0) 恕我直言更具表现力。

【讨论】:

【参考方案4】:

盗自:John Foster's answer to the question, "Need help to understand Moq better"

您可能想要测试的一件事是付款方式 当一个 65 岁以上的人被传递到 方法

[Test]
public void Someone_over_65_does_not_pay_a_pension_contribution() 

    var mockPensionService = new Mock<IPensionService>();

    var person = new Person("test", 66);

    var calc = new PensionCalculator(mockPensionService.Object);

    calc.PayPensionContribution(person);

    mockPensionService.Verify(ps => ps.Pay(It.IsAny<decimal>()), Times.Never);

【讨论】:

【参考方案5】:

在设置了Times.Never 枚举的测试之后运行验证。例如

_mock.Object.DoSomething()
_mock.Verify(service => service.ShouldntBeCalled(), Times.Never);

【讨论】:

这里的关键是,Verify(action, Never) 调用是在调用模拟之后 进行的。我以为它是为稍后调用 VerifyAll() 设置验证(这不起作用工作) 简单有效。谢谢。 不确定是不是因为版本较新,但是如果您使用“MockBehavior.Strict”设置模拟并且不添加设置,那么如果调用该方法,它将失败。无需验证。【参考方案6】:

假设你有这个方法并且你想测试它没有被调用

//Setup    
var databaseSessionMock = new Mock<IDatabaseSession>();
databaseSessionMock.Setup(m => m.Commit()).Returns(true).Verifiable();
RepositoryFactory.Configure<IDatabaseSession>(databaseSessionMock.Object);

你可以这样测试

databaseSessionMock.Verify(m => m.Commit(It.IsAny()), Times.Never(), "Database Session mock object was not used");

【讨论】:

【参考方案7】:

我意识到这是一个非常古老的问题,但它刚刚出现在我的侧边栏中,我想添加我的解决方案。

作为设置的一部分,许多单元测试似乎模拟了几个函数,但在测试期间并未使用。

当然最好启用严格模拟(这意味着任何未明确设置的内容都会引发异常),然后不要设置任何您不希望调用的函数。或者换一种说法,只设置一个测试期望调用的函数,其他的都会抛出异常。

var thingBeingTested = new Mock<IThink>(MockBehaviour.Strict);

thingBeingTested.ThisWillThrowAnExceptionBecauseItHasNotBeenMocked();

【讨论】:

以上是关于如何验证在 Moq 中没有调用该方法?的主要内容,如果未能解决你的问题,请参考以下文章

Moq - 验证除了一个方法之外没有调用任何方法

Moq:验证是否调用了一个方法

使用 Moq 来验证调用是不是以正确的顺序进行

Moq 使用对象参数验证

在 Moq 中重置模拟验证?

Moq - 不可覆盖的成员不能用于设置/验证表达式