Перейти непосредственно к среде выполнения Objective C:
#import <objc/runtime.h>
/* originally posted version — works because eventually class_getSuperclass(class)
returns nil, and class_getSuperclass(nil) does so also. */
BOOL classDescendsFromClass(Class classA, Class classB)
{
while(1)
{
if(classA == classB) return YES;
id superClass = class_getSuperclass(classA);
if(classA == superClass) return (superClass == classB);
classA = superClass;
}
}
/* shorter version; exits straight after classA turns into nil */
BOOL classDescendsFromClassShorter(Class classA, Class classB)
{
while(classA)
{
if(classA == classB) return YES;
classA = class_getSuperclass(classA);
}
return NO;
}
...
if(classDescendsFromClass(classToTest->isa, [NSObject class]) ...
class_getSuperclass
делает то, что говорит, и можно безопасно сравнивать метаклассы по указателю во время выполнения Objective C, поскольку для каждого класса существует только один экземпляр метакласса. Указатель isa - это единственное, что определенно есть в структуре objc_object.
РЕДАКТИРОВАТЬ: кроме того, в симуляторе iPhone есть известные ошибки, которые приводят к тому, что некоторые исключения не блокируются блоками try / catch. Я сообщил о них как об ошибке в Apple, и мне сказали, что у меня был дубликат, поэтому они определенно знают. Вы пробовали свой код на реальном устройстве или просто в симуляторе?
EDIT2: из более широкого контекста, приведенного в другом месте этого разговора, что-то вроде этого может быть умнее:
#import <objc/runtime.h>
BOOL classRespondsToSelector(Class classA, SEL selector)
{
return class_getInstanceMethod(classA, selector) ? YES : NO;
}
....
if(classRespondsToSelector(instance->isa, @selector(respondsToSelector:))
{
// great, we've got something that responds to respondsToSelector:; do the
// rest of our querying through there
}