self和super区别
self
是关键字
-
代表当前方法的调用者
- 如果是类方法:代表当前类
- 如果是对象方法:代表当前类的对象
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) {}
};
1
2
3
4
5
这个结构体中有两个参数:object的对象和一个superClass的结构体指针,这里的object相当于上面的self
在执行[super message]时,会做下面的事
- 编译器会先构造一个__rw_objc_super的结构体
- 然后去superClass的方法列表中找方法
- 找到之后由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.