Ответ 1
У меня также были проблемы с 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
, который является typedef из unsigned int
. Постоянная времени компиляции 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
, кстати, имеет гораздо более простую логику и ведет себя в целом как и ожидалось (опять же, посмотрите на фактический источник того, как он определяет, не выполняется ли утверждение).