最小起订量-不可覆盖的成员不得在设置/验证表达式中使用

时间:2019-07-05 15:09:37

标签: c# moq

我是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具体课程吗?只能模拟接口吗?请协助。

moq error

谢谢, 阿卜杜勒

4 个答案:

答案 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);

我不建议将此用于生产代码。事实证明,它在我的众多测试中非常有用。几年前我发现了这种方法,但最近需要再次查找,这是最热门的搜索结果。哈。天啊。