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

当我们重写父类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 类中 得到内存指针,赋值给self 参数, 在if (self){//添加A的属性 }

4.示例代码

父类QLFather初始化方法
子类QLSon初始化方法
控制器中调用

日志输入:

日志输出结果
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,776评论 0 9
  • 燕子衔泥筑暖巢, 村头老树绽花苞。 韶华珍重从今起, 未必流光把客抛。
    繁花落尽深眸阅读 342评论 8 14
  • 有孝心应该是怎么样的 从小就听着长辈说你现在要好好努力,以后长大后挣到钱就好好孝敬父母。有钱之后,买点好的东西来给...
    绯城阅读 372评论 0 1
  • 能够在最好的年纪遇到你就已经是幸运,而能够在一起,简直就是造化。原谅我恬不知耻的用了沈从文先生的话,因为我才疏学浅...
    八章十二节阅读 310评论 0 0