在单元测试的时候,有必要实力化一个组件,来让测试,简短,快速,并且隔离。
在面向对象的系统中,测试中的组件就像几个对象依赖。而不是实例化具体的类,我们使用mocks。
在测试中,Mocks是一个用预先定义好的行为,来标准化的一个假的实例对象。
在测试这些组件的时候,没什么不同。使用mocks,组件能得到很好的测试
- 常规的mock使用例子(语法是OCMock 2的语法)
Stubbed方法
下面的例子描述了stubbing语法的一般使用
<pre>
id jalopy = [OCMockObject mockForClass:[Car class]];
[[[jalopy stub] andReturn:@"75kph"] goFast:[OCMArg any] units:@"kph"]
//如果返回一个标准值,andReturnValue方法被调用
</pre>
这个例子,首先创建了一个jalopy
类实例,从class car
类。然后stub
一个方法goFaster:
该方法返回了一个NSString @"75kph"
,第一,这个stub
语法看起来有一点奇怪,但是这个一个good ideal:
ourMockObject stub] whatltShouldReturn] method
一个非常重要的提示:注意[OCMArg any]
的使用。当指定的方法需要传入一个参数,mock将会通过 andReturn
方法返回一个指定的值。当这个方法用被提供的值来调用。方法参数[OCMArg any]
告诉stub,通过任何参数都可以触发。例如
<pre>[car goFaster:84 units:@"mph"]</pre>
将永远不会触发stub因为最后的参数units:
和"kph"不匹配
Class methods 类方法
只要没有实例方法,OCMock将会寻找实例方法,在mock 实例中。
<pre>[[[[jalopy stub] classMethod] andReturn:@"expired"] checkWarrany]</pre>
mocks的类型 - niceMock
,partialMock
OCMock提供了一个不同的mocks方法,每种都有指定的使用方法
下面这种方式可以创建任何mock
<pre>id mockThing = [OCMockObject mockForClass:[Thing class]]</pre>
我称他为“vanilla”mock,如果一个没有背stub的方法被调用,vanilla mock 将会挂起。
nice mock
是友好的,如果没有stub的方法被调用,不会挂起。
<pre>Thing *someThing = [[Thing alloc] init];</pre>
最后一种类型是partial mock
.当一个没有被stub的方法被调用,方法转发到一个实际的对象。从技术上讲,这是一种欺骗,但是这是很有用的when there are aspects of the class which don’t lend themselves well to stubbing
(这个不会翻译-_-||).
<pre>Thing *someThing = [[Thing alloc] init]
id aMock = [OCMockObject partialMockForObject:someThing];
</pre>
验证方法是不是被调用,是简单的。这个可以被完成,通过,预计,拒绝和验证方法:
<pre>
id niceMockThing = [OCMock niceMockForClass[Thing class]];
[[niceMockThing expect] greeting:@"helloe"];
//验证方法想预期的被调用了
[niceMocking verify];
</pre>
如果方法没有被调用,会抛出异常。如果你使用XCTest,可以把验证方法,作为参数,放到XCTAssertNotThrow里面。Reject works the same way, but will throw when the method *is* called on the mock. Just like when stubbing, the selector and arguments passed to verify must match those passed by the caller. Use [OCMArg any] to make things easier.