美文网首页
iOS日记8-Objective-C对象模型

iOS日记8-Objective-C对象模型

作者: Mcyboy007 | 来源:发表于2018-03-27 13:43 被阅读0次

1.理解“类对象”的用意

“在运行期检查对象类型”这一操作也叫“类型信息查询”(introspection,内省)。在OC中不要直接比较对象所属的类,明智的做法是调用“类型信息查询”方法。为什么呢?从这一点,我们展开对OC对象本质的探讨。

2.什么是OC对象

首先,我们要定义对象是什么,这样我们才能更好的区分什么样的“东西”是对象。OC中的对象,具有类型信息,能接受消息。这是它最重要的内容。
从objc源码看,描述OC对象的数据结构定义如下。简单的理解,我们可以把objc_object理解为对象,把objc_class理解为类。我们可以得出,每一个对象都是一个类的实例。

typedef struct objc_class *Class;
typedef struct objc_object {
  Class isa;  //该变量定义了对象所属的类
} *id;

接着,我们再看下Class的实现,省略部分内容:

typedef stuct objc_class *Class;
struct objc_class {
  Class isa;
  Class super_class;
  const char *name;
  struct objc_method_list **methodLists;
  // ...略
};

不难看出,类(objc_class)的实现也包含了isa指针,它也能接受消息(即我们使用的类方法)。所以,我们又得出,每一个类也是一个对象。

归纳一下,OC中的类和对象都是对象,他们都含有一个isa指针,指向他们的类。并且能接受消息。

3.元类的由来

从上文可以看出类和对象都是OC对象,与之而来产生了一个问题:对象是一个类的实例,那么类是否是某一个类的实例呢?因为只有遵循这样的原则,类和对象在OC中的数据模型才能统一起来。
确实是这样的,apple的开发者为了满足这个要求,实现了“元类”,即metaclass。元类也是一个OC对象,包含了isa指针,保存了类方法的列表。
我们还可以根据继承关系图来看类、对象、元类的关系。需要注意的是,Root class(meta)super_class指向NSObject,但是它的isa指针指向了自己。

继承关系.png

4.类型信息查询

回到最开始的问题。我们知道OC中查询类型信息的方法有2种。

    1. isKindOfClass //判断出对象是否为某个特定类的实例
    1. isMemberOfClass //判断出对象是否为某类或其派生类的实例

很明显,使用“==”或者“isEqual”不能获取到更多OC对象层面的比较信息。更多的是使用类型信息查询方法,它的原理是:使用isa指针获取对象所属的类,然后通过super_class指针在继承体系中游走。这么说并不直观,我们看一下objc源码中两者的具体实现:

+ (BOOL)isKindOfClass:(Class)cls {
    for (Class tcls = object_getClass((id)self); tcls; tcls = tcls->superclass) {
        if (tcls == cls) return YES;
    }
    return NO;
}

Class object_getClass(id obj)
{
    if (obj) return obj->getIsa();
    else return Nil;
}

inline Class 
objc_object::getIsa() 
{
    if (isTaggedPointer()) {
        uintptr_t slot = ((uintptr_t)this >> TAG_SLOT_SHIFT) & TAG_SLOT_MASK;
        return objc_tag_classes[slot];
    }
    return ISA();
}

inline Class 
objc_object::ISA() 
{
    assert(!isTaggedPointer()); 
    return (Class)(isa.bits & ISA_MASK);
}

分析:
我们假设要查询的内容是Father的实例对象,即[father isKindOfClass:[NSObject Class]];
从前文说的Class的数据结构看:

  • ISA()方法其实调用的是NSObjectisa指针,即NSObjectmeta class,这和father,即Father,显然不相等。
  • 再取NSObjectmeta classsuper class,即NSObject,和father比较,显然还是不相等。
  • 再取NSObjectsuper class,即nil,和father比较,依然不相等,并停止比较。

下面我们再看看另一个比较的实现:

+ (BOOL)isMemberOfClass:(Class)cls {
    return object_getClass((id)self) == cls;
}

这个就简单很多了,单纯的拿self的isa指针的内容和cls比较。还是用[father isMemberOfClass:[NSObject Class]];来看。就是拿father,即FatherNSObject比较。显然不同。

相关文章

网友评论

      本文标题:iOS日记8-Objective-C对象模型

      本文链接:https://www.haomeiwen.com/subject/reicwttx.html