iOS中的 self = [super init] 理解

2021-08-19  本文已影响0人  小李不木

self :指向自己的指针。

super 关键字:不是指针,是一个 编译器修饰符。是编译时决定,静态的,不是在运行时确定的。

if (  self = [super init]  )    {     //  为子类进行初始化      }

防止父类的初始化方法 release 了 self  指向的空间并重新alloc了一块空间。赋值并判空可防止父类在初始化过程中发生改变,返回不同的对象。

注意:子类继承自父类,初始化子类之前要先保证父类已经初始化完毕,防止出错。当调用self = [super init] 方法时,如果父类初始化不成功,那么会返回nil,所以可以根据self是否为nil判断父类是否初始化成功,起到容错效果。

 [receiver message]:只是在编译阶段确定了要向接收者发送 message 这条消息,而 receive 将要如何响应这条消息,由运行时发生的情况来决定。

编译器会根据情况在 objc_msgSendobjc_msgSend_stret,  objc_msgSendSuper, 或 objc_msgSendSuper_stret 四个方法中选择一个调用。

如果消息是传递给父类,那么会调用名字带有”Super”的函数;

如果消息返回值是数据结构而不是简单值时,那么会调用名字带有”stret”的函数。

在 i386 平台处理返回类型为浮点数的消息时,需要用到objc_msgSend_fpret函数来进行处理,它会对浮点数寄存器做特殊处理。因为返回类型为浮点数的函数对应的 ABI(Application Binary Interface) 与返回整型的函数的 ABI 不兼容不过在 PPC 或 PPC64 平台是不需要麻烦它的。 

PS:带“Super”的是消息传递给父类;“stret”可分为“st”+“ret”两部分,分别代表“struct”和“return”;“fpret”就是“fp”+“ret”,分别代表“floating-point”和“return”。

 [self message] :调用 objc_msgSend(self, @selector(class))  

self 调用方法是从该类的方法列表当中找对应方法调用,如果没有就去父类当中找

[super message] :调用 objc_msgSendSuper(<#struct objc_super *super#>, <#SEL op, ...#>)

 super参数 :包含接收消息的实例 &  一开始寻找方法实现的父类。

 objc_super 结构体中有两个成员:

1. receiver : 消息接收者        2. super_class: 消息接收者 的父类

注意:  即由self 来调用 父类的message方法。父类message方法中出现的 self---就是调用 [super message] 方法中的self,即子类实例。

objc_msgSendSuper(<#struct objc_super *super#>, <#SEL op, ...#>) 调用流程:

1. 编译器指定一个 objc_super 结构体, 结构体中 当前类的实例对象self  为接收者。

2. 调用objc_msgSendSuper 函数,传入上述 objc_super 结构体。

3. objc_msgSendSuper 函数中直接通过偏移量直接查找到父类所在。

4. 调用  CacheLookup  函数在父类中查找方法的实现。

问题一 : [self class] & [super class] & [self  superclass] 区别:

1. [self class]  :表示当前类。self是指向自己的指针

2. [self  superclass] :表示当前类的父类

3. [super class] : 表示当前类,与调用[self class]相同,得到的永远都是self的类型

objc_super结构体:指明了消息应该被传递给特定父类。receiver仍然是self本身。当通过[super class] 获取父类时,编译器只是将指向  id 指针(self)和 SEL( class的方法选择器) 传递给了objc_msgSendSuper 函数,接着调用objc_msgSend(objc_super->receiver, @selector(class)),传入的第一个参数是指向  self  的  id指针。

问题二 : [self class] & object_getClass(self) &  object_getClass( [self class] )  的区别:

1. self 为实例对象时,[self class] 与 object_getClass(self) 等价,因为前者会调用后者。object_getClass(  [self   class]  )  得到元类。

2.  self 为类对象时,[self class] 返回值为自身,还是 self。object_getClass( self ) 与 object_getClass([self class]) 等价,得到元类。

问题三:MRC使用dealloc方法释放对象,为什么一定要调用[super dealloc] 方法?

dealloc 方法的调用顺序是从子类到父类直至 NSObject 的,NSObject 的 dealloc 会调用 object_dispose() 函数。

1:子类的很多实例变量是继承自父类的,要调用[super dealloc]方法释放从父类继承来的实例变量,继承来的这部分只能调用父类的 dealloc 方法来释放。

2:按照自下往上的逻辑,一般要先释放子类的实例,然后释放父类的实例。调用[super dealloc] 方法前如果子类中的变量未从内存中释放,容易造成子类中变量的内存积压,导致内存泄漏,[super dealloc]方法应在释放子类变量之后调用。

注意:ARC中 变量会自动释放,此方法禁止🈲⚠️调用。

dealloc 方法做了什么:

iOS dealloc方法都做了哪些东西? - 简书

iOS 底层 - dealloc 的底层实现 - 简书

参考:

iOS开发·子类初始化时为什么要调用self=[super init] - 简书

iOS:关于super 关键字(使用runtime分析) - 简书

self = [super init];_不会飞的船-CSDN博客

上一篇下一篇

猜你喜欢

热点阅读