美文网首页
self和super

self和super

作者: 雷霸龙 | 来源:发表于2021-04-02 17:40 被阅读0次

    1、self和super区别

    self
    • 是关键字
    • 代表当前方法的调用者
    • 如果是类方法:代表当前类
    • 如果是对象方法:代表当前类的对象
    super
    • 编译器指令

    2、[self message]和[super message]的实现

    其实不管是self还是super真正调用的对象都是一样的,只是查找方法的位置不一样,self是从当前类结构中开始查找,super是从父类中查找,但方法真正的接受者都是当前类或者当前类的对象

    • [self message]
      会转化为objc_msgSend(id self,SEL _cmd)这个函数,在当前类结构中找到方法并且调用

    • [super message]
      会转化为id objc_msgSendSuper(struct __rw_objc_super *super, SEL op, …)
      ,对比[self message]这里除了函数名加了super以外,第一个参数由self变成了一个结构体,下面让我们来解开这个结构体的真面目

    struct __rw_objc_super { 
        struct objc_object *object; //代表当前类的对象
        struct objc_object *superClass; 
        __rw_objc_super(struct objc_object *o, struct objc_object *s) : object(o), superClass(s) {} 
    };
    

    这个结构体中有两个参数:object的对象和一个superClass的结构体指针,这里的object相当于上面的self

    3、在执行[super message]时,会做下面的事

    • 编译器会先构造一个__rw_objc_super的结构体
    • 然后去superClass的方法列表中找方法
    • 找到之后由object调用。

    所以当你用[self Class]和[super Class]打印类的时候,打印的都是同一个类,因为他们只是查找方法的位置不同,但是调用方法的类/对象是一样的.

    4、为什么要写self = [super init]?

    因为在Xcode中,你输入init然后tab就会帮你补全这个方法,以至于我们一直都忽略了为什么在[super init]之后还要赋值给self,然后进行判断,其实这和类簇有关系,我们不能保证init的内存和alloc出来的内存是同一块内存,像NSString在alloc和init之后的对象分别是NSPlaceholderString和__NSCFConstantString*造成[super init]之后的内存被改变,所以在[super init]之后是nil,因此我们不能保证alloc和init的是同一块内存,加上这样的判断是为了提高容错性,如果init成功就返回对象,否则返回nil。

    相关文章

      网友评论

          本文标题:self和super

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