[英]“Testing Class Equality” in Objective-c
我在理解苹果指南中定义的“测试类平等”的这一部分时遇到了问题。
在动态创建的子类中,通常会重写类方法,以便子类伪装成它所替换的类。 因此,在测试类相等性时,应该比较类方法返回的值而不是低级函数返回的值。 就API而言,以下不等式适用于动态子类:
[object class] != object_getClass(object) != *((Class*)object)
因此,您应该测试两个类的相等性,如下所示:
if ([objectA class] == [objectB class]) { //...
在某些情况下,人们会在运行时添加新类。 一个例子是Key Value Observing:当你观察一个对象时,Foundation框架会创建一个被观察对象类的新子类。 此动态类的行为与其超类的行为相同,但将KVO通知添加到其所有mutator方法。
你引用的段落说,Objective-C运行时可以告诉这个新类与原始类不同。 但是,因为它只是KVO构建方式的实现细节,所以您不应该知道或关心它。 因此开发人员覆盖了他们新类的-class
方法,假装对象仍然是原始类的成员。
如果要检查两个对象是否属于同一个类,则必须比较它们的-class
方法的结果(将KVO等技巧考虑在内),而不是使用运行时函数。
这是一个例子:
#import <Foundation/Foundation.h>
#import <objc/runtime.h>
int main(int argc, const char * argv[])
{
@autoreleasepool {
NSObject *observer = [NSObject new];
NSObject *model = [NSObject new];
[model addObserver: observer forKeyPath: @"count" options: 0 context: NULL];
//using -class methods:
NSLog(@"model is a %@, observer is a %@", [model class], [observer class]);
//casting to Class:
NSLog(@"model is a %@, observer is a %@", *(Class*)model, *(Class*)observer);
//using the runtime:
NSLog(@"model is a %@, observer is a %@", object_getClass(model), object_getClass(observer));
[model removeObserver: observer forKeyPath: @"count" context: NULL];
[model release];
[observer release];
}
return 0;
}
你看到我正在做的就是创建两个对象,告诉其中一个对象,然后找出他们的类是什么。 结果如下:
2012-06-08 08:37:26.904 Untitled 2 [896:707]模型是NSObject,观察者是NSObject
2012-06-08 08:37:26.907 Untitled 2 [896:707]模型是NSKVONotifying_NSObject,观察者是NSObject
2012-06-08 08:37:26.907 Untitled 2 [896:707]模型是NSKVONotifying_NSObject,观察者是NSObject
因此,正如文档所示,它只是第一种情况(我们比较-class
),它可以完成应用程序代码可以合理预期的任何事情。 找到类的另外两种方法 - 询问运行时,并将对象指针强制转换为Class *
- 都提供了关于KVO如何从我们下面改变类的实现细节,并且意味着类比较现在不会表明班级是平等的。
因为其他答案和评论指的是-isMemberOfClass:
和-isKindOfClass:
我也将覆盖这些点:
-isKindOfClass:
不是类相等的测试。 如果object
是aClass
或其任何子类的实例,则[object isKindOfClass: aClass]
为true。 因为你引用的段落是关于类相等的, -isKindOfClass:
与此无关。 也就是说,它通常是您希望在应用程序代码中进行的测试。 关心“我可以将这个物体用作Foo
吗?”的答案更为常见。 比“这个对象真的是Foo
的实例吗?”。
-isMemberOfClass:
是类相等的测试: [object isMemberOfClass: aClass]
仅当object是aClass
的实例时才为true。 这个测试是使用-class
方法的结果完成的,这意味着在这个例子中, model
将为 [model isMemberOfClass: [NSObject class]]
测试为正。
它应该是:
if([objctA isKindOfClass [MyClass class]])
也许你可以使用'NSStringFromClass'方法并比较获得的字符串,感谢'isEqualToString'?
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.