一、self
和super
区别
self
是类,super
是预编译指令(就是给编译器看的,不是一个指针; 只要编译器看到这个标志,就会让当前对象去调用父类的方法,本质还是当前对象在调用)self
调用自己方法,super
是调用父类方法[self class]
和[super class]
输出是一样的(重写了就不一样了)
二、self和super底层实现原理
- 当使用
self
调用方法时,会从当前类的方法列表中开始找,如果没有,就从父类中再找;
当使用super
时,则从父类的方法列表中开始找,然后调用父类的这个方法.
- 当使用
self
调用时,会使用objc_msgSend
函数:id objc_msgSend(id theReceiver,SEL theSelector, ...)
.
第一个参数是消息接收者,
第二个参数是调用的具体类方法的 selector,
第三个参数是 selector 方法的可变参数.
以:
[self setName:]
为例分析:
a: 编译器会替换成调用objc_msgSend
的函数调用,
b: 其中theReceiver
是self
,theSelector
是@selector(setName:)
,这个selector
是从当前self
的class
的方法列表开始找的 setName,
c: 当找到后把对应的selector
传递过去.
- 当使用
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
的方法,将这个结构体和setName
的sel
传递过去.
b: 函数里面在做的事情类似这样:从objc_super
结构体指向的superClass
的方法列表开始找setName
的selector
,找到后再用objc_super
结构体的receiver
去调用这个selector
.
三、为什么要 self = [super init];
符合
objective-c
继承类初始化规范,super
同样也是这样.
self = [super init];
中的[super init]
是去self
的super
中调用init
;
而super
还会继续调用super
的super
的init
, 直到根类NSObject
中的init
;
根类中init
负责初始化内存区域, 向里面添加 一些必要的属性,返回内存指针,这样, 延着继承链初始化的内存指针会被从上到下传递,在不同的子类中向块内存添加子类必要的属性,直到使用的本类中得到内存指针,赋值给slef 参数.
四、class,superClass区别
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
结语
路漫漫其修远兮,吾将上下而求索~
.End
网友评论