iOS中的 self = [super init] 理解

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(message))  

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

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

 super参数 :包含接收消息的实例 &  查找方法实现的类。

 objc_super 结构体中有两个成员:

1. receiver : 消息接收者        

2. super_class:  要从这个类中查找--执行的方法message

注意:  接受者receiver 调用 super_class 类中的message方法。


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

1. 编译器指定一个 objc_super 结构体并传入,其中接收者参数:receiver 传入的是 当前类的实例对象self , receiver = self 。

2. objc_msgSendSuper 函数直接从super_class 类中查找要执行的方法。调用  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]方法应在释放子类变量之后调用,[super dealloc] 放在最后调用 

注意:ARC中 变量会自动释放,此方法禁止⚠️调用。[super dealloc] 不允许调用 

dealloc 方法做了什么:

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

iOS 底层 - dealloc 的底层实现 -


参考:

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

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

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

你可能感兴趣的:(iOS中的 self = [super init] 理解)