RhinoMocks模拟Equals方法时抛出异常

4

我有一个问题,需要为一个对象的Equals方法设置测试。

这个对象由以下接口定义:

public interface IHours {
    ITimeOfDay OpenAt { get; set; }
    ITimeOfDay CloseAt { get; set; }
    DateTime ValidFrom { get; set; }
    DateTime ValidTo { get; set; }
    bool isCovered(DateTime time);
}

它包含对ITimeOfDay的引用,定义如下:

public interface ITimeOfDay {
    DateTime Time { get; set; }
    int Hour { get; }
    int Minute { get; }
    int Second { get; }
}

现在我想要“小时等于:IHours”的值,以检查OpenAt和CloseAt属性的值。为了设置这个,我尝试将这些属性值存根化,并根据我的特定测试需要返回true和false。
    [SetUp]
    public virtual void SetUp() {
        mocks = new MockRepository();

        defaultHours = getHours();
        otherHours = getHours();

    }

    [TearDown]
    public virtual void TearDown() {
        mocks.ReplayAll();
        mocks.VerifyAll();
    }

    [Test(Description = "Equals on two Hours should regard the fields")]
    public void Equals_TwoValueEqualledObjects_Equal() {
        var openAt = mocks.Stub<ITimeOfDay>();
        var closeAt = mocks.Stub<ITimeOfDay>();

        closeAt                                   //this is line 66, referenced in the error stacktrace
            .Stub(o => o.Equals(null))
            .IgnoreArguments()
            .Return(true);

        openAt
            .Stub(c => c.Equals(null))
            .IgnoreArguments()
            .Return(true);
        mocks.ReplayAll();

        defaultHours.OpenAt = openAt;
        otherHours.OpenAt = openAt;

        defaultHours.CloseAt = closeAt;
        defaultHours.CloseAt = closeAt;

        Assert.That(defaultHours, Is.EqualTo(otherHours));
        Assert.That(defaultHours.GetHashCode(), Is.EqualTo(otherHours.GetHashCode()));
    }

但是当我运行它时,却收到这个晦涩的错误提示:
System.InvalidOperationException: Type 'System.Boolean' doesn't match the return type   'System.Collections.Generic.IList`1[NOIS.Model.Interfaces.IAircraft]' for method 'IAircraftType.get_Aircrafts();'
at Rhino.Mocks.Expectations.AbstractExpectation.AssertTypesMatches(Object value)
at Rhino.Mocks.Expectations.AbstractExpectation.set_ReturnValue(Object value)
at Rhino.Mocks.Impl.MethodOptions`1.Return(T objToReturn)
at Nois.Test.Model.Entities.HoursTest.Equals_TwoValueEqualledObjects_Equal() in HoursTest.cs: line 66 

IAircraftType接口是同一命名空间的一部分,但在测试、接口或实现类中都没有引用它。我不明白它为什么会干扰。据我所知,没有任何关于它的引用。

我正在使用 - Rhino.Mocks v3.5.0.1337 - NUnit.Framework v2.5.0.8332

有人有什么想法吗?


这是一个相当复杂的测试 - 你能否提供完整的代码?我将仔细研究,看看能否重现您的错误,但没有源代码会很困难。 - George Mauer
没关系,看我的回答 - 这实际上是一个非常有趣的案例。 - George Mauer
抱歉,我无法提供所有内容的完整源代码。我会尽量挑选必要的部分并放在一个单独的项目中。如果仍然出现奇怪的异常,我会发布它。 - Tomas
2个回答

7

这很复杂——错误很奇怪,与IAircraft无关。问题实质上在于接口不是类,因此不继承自对象。换句话说,closeAt没有Equals方法可以存根。

事实上,你甚至可以将Equals()显式调用到转换为接口的对象上,这可能有点语言上的错误。

然后有两种方法来解决这个问题:

  1. 不要模拟接口,而是模拟实现mocks.Stub()——这确实有一个虚拟的等于方法,所以你的代码将能够工作。
  2. 更好的方法是,在你的接口中添加一个Equals方法。一旦你这样做了,你就可以重写它,由于所有的类都继承自对象,你就不必明确地实现它(除非你想这样做)。

换句话说

var intrface = MockRepository.GenerateStub<IInterface>();
intrface.Stub(x => x.Equals(null)).IgnoreArguments().Return(true); 

Breaks when

public interface IInterface {
}

但是在某些情况下会起作用。
public interface IInterface {
  bool Equals(object obj);
}

我在推特上向Ayende发了这条消息,希望他能就这个疯狂的错误发表意见。 - George Mauer

0

除非我漏掉了什么,时间应该是一个简单的不可变对象。因此,我会将其实现为一个小型、经过测试的值类。然后你就可以使用真正的Equals方法。


这在这个特定的情况下是正确的。我确实做了一段时间,但是我的该死的好奇心不让我就此罢休。现在我很高兴,因为乔治上面的回答让一个糟糕的星期五变得更好了很多次。 - Tomas

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接