我是Moq的新手。我在嘲笑PagingOptions
类。这是该班级的样子:
public class PagingOptions
{
[Range(1, 99999, ErrorMessage = "Offset must be greater than 0.")]
public int? Offset { get; set; }
[Range(1, 100, ErrorMessage = "Limit must be greater than 0 and less than 100.")]
public int? Limit { get; set; }
public PagingOptions Replace(PagingOptions newer)
{
return new PagingOptions
{
Offset = newer.Offset ?? Offset,
Limit = newer.Limit ?? Limit
};
}
}
这是我的模拟课程,
var mockPagingOptions = new Mock<PagingOptions>();
mockPagingOptions.Setup(po => po.Limit).Returns(25);
mockPagingOptions.Setup(po => po.Offset).Returns(0);
设置属性值时出现以下错误。我在做错什么吗?看起来我无法Moq具体课程吗?只能模拟接口吗?请协助。
谢谢, 阿卜杜勒
答案 0 :(得分:11)
我有同样的错误,但就我而言,我试图模拟类本身而不是其接口:
// Mock<SendMailBLL> sendMailBLLMock = new Mock<SendMailBLL>(); // Wrong, causes error.
Mock<ISendMailBLL> sendMailBLLMock = new Mock<ISendMailBLL>(); // This works.
sendMailBLLMock.Setup(x =>
x.InsertEmailLog(
It.IsAny<List<EmailRecipient>>(),
It.IsAny<List<EmailAttachment>>(),
It.IsAny<string>()));
答案 1 :(得分:5)
Moq创建模拟类型的实现。如果类型是接口,则它将创建实现该接口的类。如果类型是类,则它将创建一个继承的类,并且该继承的类的成员称为基类。但是为了做到这一点,它必须覆盖成员。如果一个类的成员不能被覆盖(它们不是虚拟的,抽象的),则Moq不能覆盖它们以添加自己的行为。
在这种情况下,无需模拟PagingOptions
,因为使用真实的对象很容易。代替这个:
var mockPagingOptions = new Mock<PagingOptions>();
mockPagingOptions.Setup(po => po.Limit).Returns(25);
mockPagingOptions.Setup(po => po.Offset).Returns(0);
执行以下操作:
var pagingOptions = new PagingOptions { Limit = 25, Offset = 0 };
我们如何确定是否要嘲笑某些东西?一般而言,如果我们不想在测试中包括具体的运行时实现,则可以模拟一些东西。我们想同时测试一个课程。
但是在这种情况下,PagingOptions
只是一个保存一些数据的类。嘲笑它真的没有意义。使用真实的东西一样容易。
答案 2 :(得分:4)
我想改善Scott's的答案并给出一般答案
如果类型是一个类,则它将创建一个继承的类,并且该继承的类的成员称为基类。但是为了做到这一点,它必须覆盖成员。如果一个类的成员不能被覆盖(它们不是虚拟的,抽象的),则Moq不能覆盖它们以添加自己的行为。
在我的情况下,我必须将道具虚拟化。因此,对您的课程代码的答案是:
public class PagingOptions {
[Range (1, 99999, ErrorMessage = "Offset must be greater than 0.")]
public virtual int? Offset { get; set; }
[Range (1, 100, ErrorMessage = "Limit must be greater than 0 and less than 100.")]
public virtual int? Limit { get; set; }
public PagingOptions Replace (PagingOptions newer) {
return new PagingOptions {
Offset = newer.Offset ?? Offset,
Limit = newer.Limit ?? Limit
};
}
}
使用相同:
var mockPagingOptions = new Mock<PagingOptions>();
mockPagingOptions.Setup(po => po.Limit).Returns(25);
mockPagingOptions.Setup(po => po.Offset).Returns(0);
答案 3 :(得分:2)
如果您根据原始标题 Non-overridable members may not be used in setup / verification expressions
回答了这个问题,而其他答案都没有帮助,您可能想看看反思是否能满足您的测试需求。
假设您有一个类 Foo
,其属性定义为 public int I { get; private set; }
如果您尝试这里的答案中的各种方法,其中很少有适用于这种情况的方法。但是,您可以使用 .net 反射来设置实例变量的值,并且仍然在代码中保持相当好的重构支持。
以下是使用私有 setter 设置属性的代码段:
var foo = new Foo();
var I = foo.GetType().GetProperty(nameof(Foo.I), BindFlags.Public | BindingFlag.Instance);
I.SetValue(foo, 8675309);
我不建议将此用于生产代码。事实证明,它在我的众多测试中非常有用。几年前我发现了这种方法,但最近需要再次查找,这是最热门的搜索结果。哈。天啊。