当我们重写父类init方法时,按照苹果官方文档的要求写法为:
- (instancetype)init{
if (self = [super init]){
//写上初始化的代码
}
return self;
}
我也每次都是按照这个规范来写的,但是问我为何这样写瞬间就懵逼了,最近学习runtime,受到了启发,下面我把我的理解分享给大家,不一定完全准确,纯个人理解,欢迎斧正!
首先比较疑惑的地方:
self = [super init],看起来不就是子类self指针指向了父类super吗?
1.要搞明白这个疑惑,首先得弄清楚self,super这两个关键字的含义。
self: 代表当前方法的调用者,在对象方法中,self代表着"当前对象"; 在类方法中self代表当前类。学习过运行时都应该知道oc里面的方法底层实现对应的是函数指针(IMP),所有的函数指针都会默认自带2个参数,分别为id self和SEL _cmd;
super:仅仅只是一个“编译器指示符”,作用就是告诉编译器,去self的super类中(self的父类中)去找相应的方法,而不是父类自己去调用某方法。并不能理解super就是self的父类。superclass才是获取方法调用者父类。
2. if (self = [super init]){}都干了些什么?
if(self = [super init]){}
拆解开为
self = [super init]
if (self != nil){}
现在我们梳理一下这个流程,首先子类要重写初始化init方法,按上面的步骤写完,如何子类调用,假如这个子类叫QLSon,父类叫QLFather。
(1)QLSon *son = [QLSon alloc];
(2) son = [son init]; 通过运行时转换为:objc_msgSend(son, @selector(init));
(3) 调用这一步的时候就会进入到重写init的方法实现中,然后就会调用self = [super init];
(4) [super init]就是去调用self的父类中的init方法,通过运行时转换为:objc_msgSendSuper ( struct objc_super *super, SEL op, ... );
参数一:struct objc_super *super说明
struct objc_super {
__unsafe_unretained id receiver; //消息的实际接收者
__unsafe_unretained Class super_class; //当前类的父类
};
当我们使用super来接收消息时,编译器会生成一个objc_super结构体。这个结构体的receiver就是QLSon对象son;superClass代表QLSon的父类QLFather。
objc_msgSendSuper([son superclass], @selector(init))该函数实际的操作是:从struct objc_super结构体指向的superClass的方法列表开始查找叫init的selector,找到后再以struct objc_super结构体的receiver去调用这个selector,而此时的操作流程就是如下方式了。
objc_msgSend(objc_super->receiver, @selector(init))
由于objc_super->receiver就是self本身,所以该方法实际与下面这个调用是相同的:
objc_msgSend(self, @selector(init))
3.为什么要 self = [super init];
符合oc 继承类 初始化规范 super 同样也是这样, [super init] 去self 的super 中调用init super 调用 superSuper 的init 。直到根类 NSObject 中的init ,
根类中init 负责初始化 内存区域向里面添加 一些必要的属性,返回这样延着继承链 初始化的内存指针 被从上 到 下传递,在不同的子类中向块内存添加 子类必要的属性,直到 我们的 A 类中 得到内存指针,赋值给slef 参数, 在if (slef){//添加A的属性 }
4.示例代码
父类QLFather初始化方法子类QLSon初始化方法 控制器中调用
日志输入:
日志输出结果