我正在尝试以TDD的方式练习单元测试,所以我创建了一组我认为可以独立测试的接口。 我不确定我是否成功了,但我现在非常确定是时候使用一个模拟框架了。 我从例子中学习得最好,我认为一个非常精通的人可能能够很快将其转化为其中之一(也许是Moq或Rhino Mocks?这些是我发现的)。
单元测试:
[TestMethod]
public void RendererRendersEachWidgetByCallingBeginRenderThenEndRender()
{
// Mocks/Stubs
ISpace space = new MockSpace();
IWorld world = new MockWorld();
IWidget widget1 = new MockWidget("widget1");
IWidget widget2 = new MockWidget("widget2");
world.Add(widget1);
world.Add(widget2);
// Object under test
IRenderer target = new Renderer(world, space);
target.Render();
// verify BeginRender is called
Assert.IsTrue((space as MockSpace).results.Contains("Begin Render " + widget1.Id));
Assert.IsTrue((space as MockSpace).results.Contains("Begin Render " + widget2.Id));
// verify EndRender is called
Assert.IsTrue((space as MockSpace).results.Contains("End Render " + widget1.Id));
Assert.IsTrue((space as MockSpace).results.Contains("End Render " + widget2.Id));
// verify BeginRender is called before EndRender
var orderedResults = (space as MockSpace).results.Select((s, i) => Tuple.Create(i, s));
Assert.IsTrue(
(orderedResults.Single(x => x.Item2 == "Begin Render " + widget1.Id).Item1) <
(orderedResults.Single(x => x.Item2 == "End Render " + widget1.Id).Item1));
Assert.IsTrue(
(orderedResults.Single(x => x.Item2 == "Begin Render " + widget2.Id).Item1) <
(orderedResults.Single(x => x.Item2 == "End Render " + widget2.Id).Item1));
}
这是一个 Rhino.Mocks 版本。我在Renderer
内部未知的地方采取了自由。
如您所见,涉及相当多的设置,但在这种情况下,您不必创建所有这些设置。手动模拟对象类型。
[TestMethod]
public void RendererRendersEachWidgetByCallingBeginRenderThenEndRender()
{
// Mocks/Stubs
var orderedResultList = new List<string>();
var space = MockRepository.GenerateStub<ISpace>();
space.Stub(s => s.BeginRender(Arg<IWidget>.Is.Anything))
.WhenCalled(x => orderedResultList.Add("Begin Render " + ((IWidget)x.Arguments[0]).Id));
space.Stub(s => s.EndRender(Arg<IWidget>.Is.Anything))
.WhenCalled(x => orderedResultList.Add("End Render " + ((IWidget)x.Arguments[0]).Id));
var world = MockRepository.GenerateStub<IWorld>();
var widget1 = MockRepository.GenerateStub<IWidget>();
widget1.Stub(w => w.Id).Return("widget1"); // assuming no public setter for this property
var widget2 = MockRepository.GenerateStub<IWidget>();
widget2.Stub(w => w.Id).Return("widget2"); // assuming no public setter for this property
var widgetList = new List<IWidget>{ widget1, widget2 };
world.Stub(w => w.Widgets).Return(widgetList);
// Object under test
IRenderer target = new Renderer(world, space);
target.Render();
// verify BeginRender is called
space.AssertWasCalled(s => s.BeginRender(Arg<IWidget>.Is.Same(widget1)));
space.AssertWasCalled(s => s.BeginRender(Arg<IWidget>.Is.Same(widget2)));
// verify EndRender is called
space.AssertWasCalled(s => s.EndRender(Arg<IWidget>.Is.Same(widget1)));
space.AssertWasCalled(s => s.EndRender(Arg<IWidget>.Is.Same(widget2)));
// verify BeginRender is called before EndRender
var orderedResults = orderedResultList.Select((s, i) => Tuple.Create(i, s));
Assert.IsTrue(
(orderedResults.Single(x => x.Item2 == "Begin Render " + widget1.Id).Item1) <
(orderedResults.Single(x => x.Item2 == "End Render " + widget1.Id).Item1));
Assert.IsTrue(
(orderedResults.Single(x => x.Item2 == "Begin Render " + widget2.Id).Item1) <
(orderedResults.Single(x => x.Item2 == "End Render " + widget2.Id).Item1));
}