【问题标题】:RhinoMocks exceptions when stubbing out Equals method存根 Equals 方法时出现 RhinoMocks 异常
【发布时间】:2010-11-27 11:52:23
【问题描述】:

我在为对象上的 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; }
}

现在我想要 Equals of the Hours : IHours 检查 OpenAt 和 CloseAt IHours。为了进行设置,我尝试将这些属性值存根,然后根据我的特定测试需要它们返回 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

有人知道吗?

【问题讨论】:

  • 这是一个相当复杂的测试——你能提供完整的代码吗?我将通过这个进行挑选,看看我是否可以复制你的错误,但如果没有源代码,这将很困难
  • 没关系,看我的回答 - 这实际上是一个非常有趣的案例。
  • 我无法提供所有内容的完整来源,抱歉。我会尽量挑选出尽可能多的东西并将其放在一个单独的项目中。如果它仍然产生奇怪的异常,我会发布它。

标签: testing tdd rhino-mocks stubbing


【解决方案1】:

是的,这很复杂——错误很疯狂,这与 IAircraft 无关。本质上,问题在于接口不是类,因此不能从对象继承。换句话说 - closeAt 没有 Equals 方法来存根。

事实上,它可能有点语言缺陷,您甚至可以在转换为接口的对象上显式调用 Equals()。

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

  1. 不要模拟接口,模拟实现 mocks.Stub() - 这确实有一个虚拟的 equals 方法,因此您的代码可以工作。
  2. 更好的是,在您的界面中添加一个 Equals 方法。一旦你这样做了,你将能够覆盖它,并且由于所有类都继承自对象,你将不必显式地实现它(除非你愿意)。

换句话说

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

中断时

public interface IInterface {
}

但工作时

public interface IInterface {
  bool Equals(object obj);
}

【讨论】:

  • 我在推特上把这个发给了 Ayende,希望他能对这个 carzy 错误发表意见
【解决方案2】:

除非我遗漏了什么,否则 Time of Day 应该是一个简单的不可变对象。所以我只是将它实现为一个小的、经过测试的值类。然后就可以使用真正的 Equals 方法了。

【讨论】:

  • 在这种特殊情况下确实如此。我确实这样做了一段时间,但我的好奇心不会让我离开它。我现在很高兴,因为乔治在上面的回答让可怕的星期五好很多倍。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2021-02-15
  • 2010-12-15
  • 1970-01-01
  • 2011-06-03
  • 2016-01-12
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多