Objective-C与Swift语言对象的初始化

探究一下Objective-C与Swift语言对象的初始化。

一些概念的定义(Objective-C和Swift定义类似,这里以Swift为例)

1.指定初始化器( Designated Initializers)
指定初始化器是指覆盖面最广的初始化器,保证父类被继承过来的变量被调用初始化。(A designated initializer fully initializes all properties introduced by that class and calls an appropriate superclass initializer to continue the initialization process up the superclass chain.

init(parameters) {
    statements
}

2.便利初始化器(Convenience Initializers)
便利初始化器是指第二重要的初始化器,不是覆盖面最广的初始化器,但是,一定会调用指定初始化器的初始化器。(You can define a convenience initializer to call a designated initializer from the same class as the convenience initializer with some of the designated initializer’s parameters set to default values.

convenience init(parameters) {
    statements
}

Tips:
1.每个类必须要有一个或者多个指定初始化器。
2.便利初始化器可以有多个。

规则:(Objective-C有点不一样,成员变量会默认初始化为nil或者0)
规则1: 指定初始化器必须调用直接父类的指定初始化器。
规则2: 便利初始化器必须调用类的另外一个初始化器(包括指定初始化器和便利初始化器)。
规则3: 便利初始化器必须最终调用类的指定初始化器。

如下图:


initializerDelegation.png

Objective-C对象的初始化

Objective-C对象的初始化分为两个函数alloc和init。顾名思义,alloc函数是负责初步的内存分配,init函数负责变量的初始化。但是,苹果官方建议alloc和init函数要一起调用。因为,有可能init后会返回nil或者不一样的对象地址。因为init可以返回nil也可以返回另一个替换对象,所以一切要以init为准。

实现一个初始化器需要注意的要点:
1.必须先调用父类初始化器。
2.检查父类返回的对象。如果为nil,则不能执行该类的初始化实体,直接返回nil。
3.初始化引用对象实例,根据需要retain或者copy对象。
4.在设置完实例变量为可用的初始值时,返回self引用,除非:
(1)需要返回一个替代的对象。
(2)一些别的原因需要返回nil。

init_inheritance_chain.png

Swift对象的初始化

Swift初始化分为两个阶段:
第一个阶段,保证每一个储存属性有初始值。第二个阶段,可以进一步定制化储存属性。

第一阶段:
1.指定或者便利初始化函数被调用。
2.为新的实例分配内存空间。此时,所分配的内存还没有初始化。
3.指定初始化器确保所有的储存属性有初始值。现在,这些储存属性所分配的内存空间已初始化。
4.指定初始化器调用父类的指定初始化器,为父类的储存属性执行同样的操作。
5.这样的操作,沿着继承链一直到根类。
6.一旦到达根类,确保最后一个类所有的储存属性已经初始化,对象的内存被认为完全初始化,第一阶段结束。

第二阶段:
1.从继承链的顶部往回走,继承链中的所有指定初始化器
都有选择定制化实例的机会。初始化器现在已经可以访问self和修改实例的属性,调用实例方法等等。
2.最后,在继承链上的任意便利初始化器都有机会去定制化实例,并且调用self去做一些事。

Swift的编译器为保证两个阶段的正常进行,进行了一些安全检查:
安全检查1:
指定初始化器在调用父类初始化器之前,必须保证本类所有的储存属性都有值。满足第一阶段的第三步。
安全检查2:
指定初始化器在赋值给继承的属性之前,必须调用父类的指定初始化器。防止父类赋值覆盖。
安全检查3:
一个便利初始化器在给任意属性(包括本类的属性和继承的属性)赋值之前,必须调用另外一个初始化器(包括便利初始化器和指定初始化器)。防止指定初始化器属性赋值覆盖便利初始化器。
安全检查4:
一个初始化器在第一阶段没有完成前,不能调用任意实例方法,不能访问任意的实例属性或者引用self。因为在第一阶段没有完成前,该实例还不是完全可用的。

Phase1.png
Phase2.png

初始化器的继承和重写
Swift和Objective-C不同,子类默认不继承父类的初始化器。因为如果子类默认继承父类的初始化器,那么,如果子类在调用父类的指定初始化时,子类自定义的一些实例变量还没有完全初始化就会报错,Swift就是防止会在无意之间发生的错误。但是,Swift规定在某些环境下子类会继承父类的初始化器。如果子类需要重写父类的初始化器,需要在初始化器前面添加override的关键字。

上面提到Swift的子类会在某些环境下继承父类的初始化器,下面有两个规则:
规则1:
如果子类没有定义任何指定初始化器,子类会自动继承父类所有的指定初始化器。(子类没有定义任何指定初始化器,那么意味着会用默认的初始化器,意味着所有的自定义变量都已经初始化,所以,这时候直接调用父类的初始化器,不会造成子类自定义的变量还没有初始化的情况。)
规则2:
如果子类提供类所有父类的指定初始化器的实现,包括从规则继承而来的或者自定义实现所有父类的指定初始化器,那么,子类自动继承父类的所有便利初始化器。(重写和继承父类的指定初始化器都可以保证调用父类的指定初始化时,子类自定义的所有变量已经初始化完成。)

Tips:
1.子类可以实现父类指定的初始化器并把它当作子类的便利初始化器。
2.子类如果重写父类的便利初始化器,则父类的便利初始化器永远也不能被调用。

可失败初始化器
Swift和Objective-C不同,初始化器不用返回值,默认都是成功的。可失败初始化器,顾名思义也就是可以返回nil,相对于Swift来说就是返回optional类型。如下:

init?(parameters) {
    if something == nil {
        return nil
    }
    ...
}

重写可失败初始化器
和别的初始化器一样,可失败初始化器可以被重写。并且,子类可以用不可失败的初始化器重写父类可失败初始化器

必须初始化器
如果你想要子类必须实现父类的初始化器,那么,需要在父类的初始化器的前面添加required的关键字,如下:

class  SomeClass {
    required init() {
    }
}

class SomeSubclass: SomeClass {
    required init() {
    }
}

总结

上面总结了Objective-C与Swift语言对象的初始化的一些关键点和一些注意的地方。这也是作为笔记记录,忘记的时候过来翻翻。

参考:

Concepts in Objective-C Programming
The Swift Programming Language (Swift 4.0.3)
Objective-C对象之初始化和两段构造法(二)

你可能感兴趣的:(Objective-C与Swift语言对象的初始化)