self :指向自己的指针。
super 关键字:不是指针,是一个 编译器修饰符。是编译时决定,静态的,不是在运行时确定的。
if ( self = [super init] ) {
// 为子类进行初始化
}
赋值并判空可防止父类在初始化过程中发生改变,返回不同的对象。
防止父类的初始化方法 release 了 self 指向的空间,并重新alloc分配一块新的内存空间。
注意:子类继承自父类,初始化子类之前要先保证父类已经初始化完毕,防止出错。当调用self = [super init] 方法时,如果父类初始化不成功,那么会返回nil,所以可以根据self是否为nil判断父类是否初始化成功,起到容错效果。
[receiver message]:只是在编译阶段确定了要向接收者发送 message 这条消息,而 receive 将要如何响应这条消息,由运行时发生的情况来决定。
编译器会根据情况在 objc_msgSend, objc_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博客