ios开发self和super的区别

本文深入解析了Objective-C中self和super的区别与联系。self代表当前方法的调用者,而super用于从父类中查找方法。文章详细解释了两者如何在运行时被转换成不同的函数调用,并探讨了在初始化过程中使用self=[super init]的原因。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

self和super区别

self

  1. 是关键字
  2. 代表当前方法的调用者
    • 如果是类方法:代表当前类
    • 如果是对象方法:代表当前类的对象

super

编译器指令

[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
在执行[super message]时,会做下面的事
1. 编译器会先构造一个__rw_objc_super的结构体
2. 然后去superClass的方法列表中找方法
3. 找到之后由object调用。
所以当你用[self Class]和[super Class]打印类的时候,打印的都是同一个类,因为他们只是查找方法的位置不同,但是调用方法的类/对象是一样的.

为什要写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.