self = [super init]

作者: iOS开发章鱼哥 | 来源:发表于2015-07-14 09:53 被阅读2367次

初始化方法的标准结构是这样子的:

- (instancetype)init
{
    self = [super init]; // call the designated initializer
    if (self) {
        // Custom initialization
    }
    return self;
}

我们主要来看看,这一句:

    self = [super init]; // call the designated initializer

问题1:[super init] 到底做了什么?

问题2:为什么把[super init]的值赋值给self?

问题3:我们用过了 alloc init的两部创建,为什么不是[[super alloc] init]

先来回答问题3,因为这个问题是来搞笑的:

我们看下alloc方法是怎么定义的 
+ (instancetype)alloc;
可以看到alloc是一个类方法,而super不是一个类.

然后我们在来看看问题1:

[super init] 的结果可能有三种:
第一种:  [super init] 初始化成功,这个是最最正常的情况。
第二种:    [super init] 初始化失败,我们都知道oc的两步创建,alloc开辟内存空间,init初始化对象,但是init还有另外一个作用,在初始化失败的时候,init方法会返回一个nil。回头行文的最上面,如果self为nil,那么if(self)语句的内容,将不被执行,已确保程序健壮安全。
- (instancetype)init
{
    self = [super init]; // call the designated initializer
    if (self) {
        // Custom initialization
    }
    return self;
}
第三种:self = [super init]执行之后,内存指向了不相关的地址。这种情况的出现,一般是一下几种情况:单例、类簇、对象为NSNumber类型、父类在初始化的时候释放了当前对象,然后重新开辟了新的内存地址。   

最后,我们来使用《禅与 Objective-C 编程艺术》的原话来回答问题2:

如果你的超类没有成功初始化它自己,你必须假设你在一个矛盾的状态,并且在你的实现中不要处理你自己的初始化逻辑,同时返回 nil。如果你不是这样做,你看你会得到一个不能用的对象,并且它的行为是不可预测的,最终可能会导致你的 app 发生 crash。

一句话概括:

if( self = [super init] )这是一种通常的建议写法,赋值并测试nil只是为了防止超类在初始化过程中发生改变,返回了不同的对象。

参考:

《禅与 Objective-C 编程艺术》中文版
中文版地址:https://github.com/oa414/objc-zen-book-cn#class-cluster

相关文章

网友评论

本文标题:self = [super init]

本文链接:https://www.haomeiwen.com/subject/tsbuqttx.html