美文网首页
iOS - 关键字self和super

iOS - 关键字self和super

作者: 搬砖的crystal | 来源:发表于2022-11-08 10:40 被阅读0次

    一、selfsuper 区别

    1. self 是类,super 是预编译指令(就是给编译器看的,不是一个指针; 只要编译器看到这个标志,就会让当前对象去调用父类的方法,本质还是当前对象在调用)
    2. self 调用自己方法,super 是调用父类方法
    3. [self class][super class] 输出是一样的(重写了就不一样了)

    二、selfsuper底层实现原理

    1. 当使用 self 调用方法时,会从当前类的方法列表中开始找,如果没有,就从父类中再找;
      当使用 super 时,则从父类的方法列表中开始找,然后调用父类的这个方法。
    2. 当使用 self 调用时,会使用 objc_msgSend 函数:id objc_msgSend(id theReceiver,SEL theSelector, ...)
      第一个参数是消息接收者,第二个参数是调用的具体类方法的 selector,第三个参数是 selector 方法的可变参数。
      (以: [self setName:] 为例分析:
      a: 编译器会替换成调用 objc_msgSend的函数调用,
      b: 其中 theReceiverselftheSelector@selector(setName:),这个 selector 是从当前 selfclass 的方法列表开始找的 setName
      c: 当找到后把对应的 selector传递过去。)
    3. 当使用 super 调用时,会使用 objc_msgSendSuper 函数:id objc_msgSendSuper(struct objc_super *super,SEL op, ...)
      第一个参数是个 objc_super 的结构体,
      第二个参数还是类似上面的类方法的 selector
    struct objc_super {
        id receiver;
        Class superClass; 
    };
    

    (以: [super setName:]为例分析:
    a: 构建 objc_super 的结构体,此时这个结构体的第一个成员变量 receiver 就是子类,和 self 相同。而第二个成员变量 superClass 就是指父类调用 objc_msgSendSuper 的方法,将这个结构体和 setNamesel 传递过去
    b: 函数里面在做的事情类似这样:从 objc_super 结构体指向的 superClass 的方法列表开始找 setNameselector,找到后再用 objc_super 结构体的 receiver 去调用这个 selector

    三、为什么要 self = [super init];

    符合 objective-c 继承类初始化规范, super 同样也是这样。
    self = [super init] 中的 [super init] 是去 selfsuper 中调用 init,而 super 还会继续调用 supersuperinit, 直到根类 NSObject 中的 init
    根类中 init负责初始化内存区域, 向里面添加 一些必要的属性,返回内存指针,这样, 延着继承链初始化的内存指针会被从上到下传递,在不同的子类中向块内存添加子类必要的属性,直到使用的本类中得到内存指针,赋值给 slef 参数。

    四、classsuperClass区别

    class获取当前方法调用者的类
    superclass获取当前方法调用者的父类

    #import "Son.h"
    
    @interface Son : Person
    
    @end
    
    @implementation Son
    
    - (instancetype)init {
        self = [super init];
        if (self) {
            NSLog(@"%@", NSStringFromClass([self class]));       // 获取当前方法调用者的类 (Son)
            NSLog(@"%@", NSStringFromClass([super class]));      // 本质还是当前方法调用者 (Son)
            NSLog(@"%@", NSStringFromClass([self superclass]));  // 当前方法调用者的父类 (Person)
            NSLog(@"%@", NSStringFromClass([super superclass])); // 当前方法调用者的父类 (Person)
        }
        return self;
    }
    
    @end
    

    相关文章

      网友评论

          本文标题:iOS - 关键字self和super

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