这是摘自Objective-C runtime programming guide
:
创建新对象时,会为其分配内存,并初始化其实例变量。对象变量中的第一个是指向其类结构的指针。这个名为isa的指针使对象可以访问它的类,并通过该类访问它继承的所有类。
isa在NSObject
中声明如下:
Class isa;
反过来,Class
只不过是指向结构
typedef struct objc_class *Class;
现在让我们来看看这个结构:
struct objc_class {
Class isa;
#if !__OBJC2__
Class super_class OBJC2_UNAVAILABLE;
const char *name OBJC2_UNAVAILABLE;
long version OBJC2_UNAVAILABLE;
long info OBJC2_UNAVAILABLE;
long instance_size OBJC2_UNAVAILABLE;
struct objc_ivar_list *ivars OBJC2_UNAVAILABLE;
struct objc_method_list **methodLists OBJC2_UNAVAILABLE;
struct objc_cache *cache OBJC2_UNAVAILABLE;
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE;
#endif
}
我们可以看到指向超类的指针(以及结构的所有其余成员,除了另一个成员之外的isa)在最新版本的Objective-C中不可用。
所以我的问题是,如果super_class
指针不可用,对象如何才能访问其超类?是否可以通过另一个isa指针访问超类?但究竟是怎么回事?这个怎么运作?有人能解释一下吗?
答案 0 :(得分:7)
刚检查了源文件
Class class_getSuperclass(Class cls)
{
return _class_getSuperclass(cls);
}
#define newcls(cls) ((class_t *)cls)
Class
_class_getSuperclass(Class cls)
{
return (Class)getSuperclass(newcls(cls));
}
static class_t *
getSuperclass(class_t *cls)
{
if (!cls) return NULL;
return cls->superclass;
}
所以Class
实际上是指向class_t
typedef struct class_t {
struct class_t *isa;
struct class_t *superclass;
Cache cache;
IMP *vtable;
uintptr_t data_NEVER_USE; // class_rw_t * plus custom rr/alloc flags
class_rw_t *data() const {
return (class_rw_t *)(data_NEVER_USE & ~(uintptr_t)3);
}
void setData(class_rw_t *newData) {
uintptr_t flags = (uintptr_t)data_NEVER_USE & (uintptr_t)3;
data_NEVER_USE = (uintptr_t)newData | flags;
}
bool hasCustomRR() const {
#if CLASS_FAST_FLAGS_VIA_RW_DATA
return data_NEVER_USE & (uintptr_t)1;
#else
return data()->flags & RW_HAS_CUSTOM_RR;
#endif
}
void setHasCustomRR(bool inherited = false);
bool hasCustomAWZ() const {
#if CLASS_FAST_FLAGS_VIA_RW_DATA
return data_NEVER_USE & (uintptr_t)2;
#else
return data()->flags & RW_HAS_CUSTOM_AWZ;
#endif
}
void setHasCustomAWZ(bool inherited = false);
bool isRootClass() const {
return superclass == NULL;
}
bool isRootMetaclass() const {
return isa == this;
}
} class_t;
这是保存所有内容的结构
无论如何,这些都是内部实施细节,不应该依赖。所以不要编写依赖于这些的代码,因为它们可能会在下次运行时更新时破坏
答案 1 :(得分:1)
不依赖于类struct的任何内部。 - 你也不依赖其他好友实例变量!你寻找属性
您可以安全地使用运行时在runtime.h
获取超类eg.call class_getSuperclass