您的当前位置:首页对重写初始化方法时self = [super init]的写法解

对重写初始化方法时self = [super init]的写法解

2024-12-14 来源:哗拓教育

当我们重写父类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初始化方法 控制器中调用

日志输入:

日志输出结果
显示全文