我在 Xcode 5 的测试中也遇到了不少麻烦。它似乎仍然存在一些奇怪的行为,但我已经找到了你的特定 XCTAssertEqual 不起作用的明确原因。
如果我们看一下测试代码,我们会发现它实际上做了以下事情(直接取自 XCTestsAssertionsImpl.h - 在那里可能更容易查看):
#define _XCTPrimitiveAssertEqual(a1, a2, format...) \
({ \
@try { \
__typeof__(a1) a1value = (a1); \
__typeof__(a2) a2value = (a2); \
NSValue *a1encoded = [NSValue value:&a1value withObjCType:@encode(__typeof__(a1))]; \
NSValue *a2encoded = [NSValue value:&a2value withObjCType:@encode(__typeof__(a2))]; \
float aNaN = NAN; \
NSValue *aNaNencoded = [NSValue value:&aNaN withObjCType:@encode(__typeof__(aNaN))]; \
if ([a1encoded isEqualToValue:aNaNencoded] || [a2encoded isEqualToValue:aNaNencoded] || ![a1encoded isEqualToValue:a2encoded]) { \
_XCTRegisterFailure(_XCTFailureDescription(_XCTAssertion_Equal, 0, @#a1, @#a2, _XCTDescriptionForValue(a1encoded), _XCTDescriptionForValue(a2encoded)),format); \
} \
} \
@catch (id exception) { \
_XCTRegisterFailure(_XCTFailureDescription(_XCTAssertion_Equal, 1, @#a1, @#a2, [exception reason]),format); \
}\
})
问题出在这里:
测试实际所做的是将值编码为NSValue,然后比较它们。 “好吧,”你说,“但那有什么问题?”在我为它制作自己的测试用例之前,我也不认为有一个。问题是 NSValue 的 -isEqualToValue 还必须比较 NSValue 的 编码类型 以及它的实际值。 两者必须相等才能返回YES。
在您的情况下,arr.count 是 NSUInteger,它是 unsigned int 的 typedef。编译时常量3 可能在运行时退化为signed int。因此,当将两者放入NSValue 对象时,它们的编码类型不相等,因此根据-[NSValue isEqualToValue],两者不能相等。
您可以通过一个自定义示例来证明这一点。以下代码明确执行 XCTAssertEqual 所做的事情:
// Note explicit types
unsigned int a1 = 3;
signed int a2 = 3;
__typeof__(a1) a1value = (a1);
__typeof__(a2) a2value = (a2);
NSValue *a1encoded = [NSValue value:&a1value withObjCType:@encode(__typeof__(a1))];
NSValue *a2encoded = [NSValue value:&a2value withObjCType:@encode(__typeof__(a2))];
if (![a1encoded isEqualToValue:a2encoded]) {
NSLog(@"3 != 3 :(");
}
"3 != 3 :("每次都会出现在日志中。
我赶紧在这里补充一下,这实际上是预期的行为。 NSValue应该在进行比较时检查其类型编码。不幸的是,这不是我们在测试两个(“相等”)整数时所期望的。
XCTAssertTrue 顺便提一下,它的逻辑要简单得多,并且行为通常符合预期(再次,请参阅实际来源,了解它如何确定断言是否失败)。