Mock Object工具横向比较(下)

本文介绍了EasyMock和MockCreator两种测试工具。EasyMock是功能丰富、使用灵活的动态Mock工具,可检查方法调用次序等;MockCreator能为Class自动生成Mock Object实现,可通过命令行或Eclipse插件操作。二者编写测试方法模式相同,可按需选用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

EasyMock

EasyMock自己的话来说:EasyMock is a class library that provides an easy way to use Mock Objects for given interfacesEasyMock提供了功能丰富的Mock Object。可以检查方法调用的次序,可以设定某个特定方法调用次数的范围,甚至可以不检查对Mock Object方法调用的一致性。可以说,EasyMock的动态Mock工具中使用最灵活的一个。

下面,来看看它的测试方法:

 

1.      public void testSimpleAdditionUsingEasyMock() throws Exception {

2.       MockControl control1 = MockControl.createControl(HttpServletRequest.class);

3.       HttpServletRequest mockHttpServletRequest =

4.               (HttpServletRequest)control1.getMock();

5.       MockControl control2 = MockControl.createStrictControl(HttpServletResponse.class);

6.       HttpServletResponse mockHttpServletResponse =

7.               (HttpServletResponse)control2.getMock();

8.

9.       mockHttpServletRequest.getParameter("b");

10.      control1.setReturnValue("4", 1);

11.      mockHttpServletRequest.getParameter("a");

12.      control1.setReturnValue("3", 1);

13.

14.      final StringWriter output = new StringWriter();

15.      final PrintWriter contentWriter = new PrintWriter(output);

16.

17.      mockHttpServletResponse.setContentType("text/html");

18.      control2.setVoidCallable(1);

19.      mockHttpServletResponse.getWriter();

20.      control2.setReturnValue(contentWriter, 1);

21.

22.      control1.replay();

23.      control2.replay();

24.

25.      SimpleCalcServlet aServlet = new SimpleCalcServlet();

26.      aServlet.doGet(mockHttpServletRequest, mockHttpServletResponse);

27.

28.      control1.verify();

29.      control2.verify();

30.      assertEquals("Output should be an addition", "Result = 7", output.toString());

31.      }

 

可以看出,代码样式与MockObjects是完全一样的,只是细节上略有不同。

Line2Line5创建MockControl时,Line2创建的是一个不检查方法调用次序的ControllerLine5创建了一个检查方法调用次序的Controller。然后调用ControllergetMock()取得Mock实例。

Line9mockHttpServletRequest的训练中,直接调用了将要被被测代码调用的方法;然后Line10设定了调用这个方法的返回值和调用次数。最后,Line22Line23调用Controllerreplay()方法,表示训练结束,可以开始测试了。

MockCreator

MockObjectsEasyMock只能为接口(Interface)动态创建Mock Object实例,这对于已经养成了使用接口这个好习惯的我们来说,固然没什么好抱怨的J。然而世事不尽人意,更多的时候我们不得不为一些Class编写Mock实现,这时,类似MockCreator这一类能自动生成Mock Object实现的工具就派上用场了。

MockCreator可以用命令行形式生成Mock Object Class,也可以使用其官方提供的插件在Eclipse中方便的操作。下图就是使用MockCreator插件在Eclipse中生成Mock Object Class的操作方法:

 

 

对于上图所示的例子来说,自动生成的Mock Object Class的类名为MockHttpServletRequest,与HttpServletRequest放在同一个包下。

怎么样,非常简单就能生成一个Mock Object实现了,不是吗?J

再来看看利用它写的测试方法:

 

1.      public void testSimpleAdditionUsingMockCreator() throws Exception {

2.       final MockHttpServletRequest mockHttpServletRequest = new MockHttpServletRequest();

3.       final MockHttpServletResponse mockHttpServletResponse =

4.               new MockHttpServletResponse();

5.

6.      mockHttpServletRequest.startBlock();

7.       mockHttpServletRequest.expectGetParameter("b", "4");

8.       mockHttpServletRequest.expectGetParameter("a", "3");

9.       mockHttpServletRequest.endBlock();

10.

11.      final StringWriter output = new StringWriter();

12.      final PrintWriter contentWriter = new PrintWriter(output);

13.

14.      mockHttpServletResponse.expectSetContentType("text/html");

15.      mockHttpServletResponse.expectGetWriter(contentWriter);

16.

17.      SimpleCalcServlet aServlet = new SimpleCalcServlet();

18.      aServlet.doGet(mockHttpServletRequest, mockHttpServletResponse);

19.

20.      mockHttpServletRequest.verify();

21.      mockHttpServletResponse.verify();

22.      assertEquals("Output should be an addition", "Result = 7", output.toString());

23.  }

 

请注意,在Line6Line9分别调用了startBlock()endBlock()方法,这表示在最后验证的时候,不对Block内的方法调用次序进行检查。Line7Line8表示被测代码将分别以参数“b”、“a”调用getParameter()方法,届时mockHttpServletRequest将返回“4”、“3”。

总结

从上面的代码可以看出,使用这三个工具编写test方法的模式都完全一样,功能也基本相同,仅是他们提供的用户接口略有不同。

EasyMock虽然接口稍有复杂,但是使用起来也最为灵活,在测试一些递归等循环次数较多的方法时有独到之处。

MockObjectsMockCreator接口简单易用。但一个是动态Mock,一个是静态Mock,可以根据实际情况决定采用其中的哪个。

另外,在需要为Class建立Mock Object的情况下,就只有MockCreator能承此大任了。

最重要的,根据项目的实际需要,我们也可以同时采用多个工具来帮助我们进行测试,以满足不同的需求。

参考资料

1、  www.mockobjects.comMock Object Project的老家,致力于建立Mock Object标准框架。

2、  www.easymock.orgEasyMock,致力于提供一种简单的途径来帮助创建Mock Object

3、  mockcreator.sf.netMockCreator,自动生成Mock Object,提供了易用的Eclipse插件。

4、  www.mockmaker.orgMockMaker,与MockCreator类似的Mock Object自动生成工具。因在某些环境下不能正确的生成Mock Object实现,所以被排除在了此次比较之外。

### 使用 Mock ObjectMocking Framework 进行单元测试的具体实现 #### 什么是 Mock ObjectMock 对象是一种用于替代实际依赖的对象,在单元测试中用来模拟外部系统的交互行为。它允许开发者隔离待测代码,从而专注于验证核心逻辑而无需考虑真实的外部环境。 --- #### Java 中使用 Mockito 实现 Mock Object 在 Java 生态中,`Mockito` 是最流行的 mocking framework 工具之一。以下是其基本用法: 1. **创建 Mock 对象** 可以通过 `@Mock` 注解或者 `mock()` 方法来创建一个 mock 对象。 ```java @Mock private MyDependency myDependency; // 或者手动创建 MyDependency myDependency = mock(MyDependency.class); ``` 2. **设置 Mock 行为** 使用 `when(...).thenReturn(...)` 定义 mock 对象的行为。 ```java when(myDependency.getUser()).thenReturn(new User("John")); ``` 3. **注入到被测类** 将 mock 对象注入到被测类中,通常可以通过构造函数或 setter 方法完成。 ```java MyClassToTest classUnderTest = new MyClassToTest(myDependency); ``` 4. **执行测试并断言** 调用被测方法后,利用 JUnit 提供的断言工具验证结果是否符合预期。 ```java User result = classUnderTest.getUser(); assertNotNull(result); assertEquals("John", result.getName()); verify(myDependency, times(1)).getUser(); // 验证 getUser() 是否只调用了 1 次 ``` 上述过程展示了如何基于 Mockito 创建和配置 Mock 对象,并结合 JUnit 断言进行单元测试[^2]。 --- #### C# 中使用 Moq 实现 Mock Object C# 开发环境中常用的是 `Moq` 库作为 mocking framework。以下是一个简单的例子: 1. **安装 NuGet 包** 确保项目已安装 `Moq` 包。 2. **定义接口及其 Mock** 利用 `Mock<T>` 类型创建一个 mock 对象。 ```csharp var mockContext = new Mock<MyDbContext>(); var mockUser = new Mock<User>(); mockContext.Setup(c => c.User).Returns(mockUser.Object); ``` 3. **集成到测试场景** 将 mock 上下文传递给被测类实例化。 ```csharp var service = new UserService(mockContext.Object); var userResult = service.GetUserById(1); Assert.AreEqual("ExpectedName", userResult.Name); mockContext.Verify(c => c.User, Times.Once()); // 验证上下文中 User 属性访问次数 ``` 此示例说明了如何借助 Moq 设置复杂对象之间的关系以及验证它们间的互动情况[^1]。 --- #### Golang 中使用 gomock 实现 Mock Object 对于 Go 编程语言而言,`gomock` 是一种强大的 mocking 解决方案。下面是它的典型应用流程: 1. **安装 gomock 工具链** 执行命令安装必要的二进制文件: ```bash go install github.com/golang/mock/mockgen@latest ``` 2. **生成 Mock 文件** 根据目标接口自动生成对应的 mock 结构体。 ```bash mockgen -source=path/to/your_interface.go -destination=mocks/mock_your_interface.go -package=mocks ``` 3. **编写测试案例** 导入生成好的 mock 包,并设定期望行为。 ```go import ( "testing" . "path/to/mocks" ) func TestMyFunction(t *testing.T) { ctrl := gomock.NewController(t) defer ctrl.Finish() mockDep := NewMockYourInterface(ctrl) mockDep.EXPECT().DoSomething(gomock.Any()).Return(nil) sut := YourService{Dependency: mockDep} err := sut.CallMethod() if err != nil { t.Errorf("Unexpected error occurred: %v", err) } } ``` 这段代码片段体现了如何运用 gomock 控制依赖项的动作路径,进而保障服务层功能正常运行[^3]。 --- #### 总结 无论是在哪种编程语言环境下,mock object 的主要目的都是为了简化对外部资源的管理难度,使得单个模块能够独立于其他部分接受严格审查。以上分别介绍了三种主流技术栈下的实践技巧,希望对你有所帮助! ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值