Swift学习:构造器(中)
Swift之构造器(上)
Swift之构造器(中)
Swift之构造器(下)
本篇继续对Swift中的构造器进行介绍,这里主要说到类的继承和构造。 作为引用类型的类具有的继承特性,这使得类的构造过程较为复杂一些,因为涉及到很多细节性的问题。在使用之前,我们需要了解一些基本的问题:
- 类中所有存储属性,包括继承父类的属性,都要在构造过程中设置初值
- Swift类的构造器分为指定构造器和便利构造器,确保完成构造过程
一、指定构造器和便利构造器
指定构造器:
- 类的主要构造器,负责初始化类中所有属性,在继承关系中可调用父类链中的父类构造器
- 每个类至少一个指定构造器。但是某些情况下,许多类是通过继承父类的指定构造器来满足这个条件
便利构造器:
- 辅助类型的构造器,调用同一个类中的指定构造器完成类的初始化操作
- 便利构造器需要在init关键字之前添加convenience关键字,使用空格分开
class FatherClass {
var valueOne: Int
//指定构造器
init(valueOne: Int){
self.valueOne = valueOne
}
}
class ChildClass: FatherClass {
var valueTwo:Int //子类的新引入属性
//指定构造器
init(valueOne: Int , valueTwo:Int){
self.valueTwo = valueTwo
//先初始化新引入属性,再初始化超类
super.init(valueOne: valueOne)
}
//便利构造器
override convenience init(valueOne: Int){
self.init(valueOne: valueOne);
}
}
二、类的构造器的使用
关于类的构造器的使用在使用的时候,细节方面需要要注意很多,通过查找一些资料,现总结如下:
1. 指定构造器和便利构造器的基本使用原则:
规则1:当前类存在父类时,指定构造器器必须调用其直接父类的指定构造器,为保证继承的属性得以初始化。
规则2:便利构造器必须调用同一类中定义的其他构造器。而且最终会导致一个指定构造器被调用。
2. Swift两段式构造过程
Swift 中类的构造过程包含两个阶段,被称为是二段式构造
第一阶段:每个存储型的值指定一个初值。
第二阶段:给当前类一个机会,在新实例准备使用之前进一步修改定制存储型的属性。
与OC的区别:Swift与OC的构造过程相似,区别在于阶段一,OC给每个属性赋值为0或者空值,但是Swift更为灵活,允许开发者指定自己所需的初始值(默认值)。
3. 两段式构造需要注意的事项
- 指定构造器必须保证当前类新引入所有属性初始化完成之后,才能将其构造器任务继续向上代理给父类的构造器。
- 指定构造器必须先向上代理调用父类构造器,然后再为其继承的属性设置新值,否则,指定构造器赋予的新值将被父类中的构造器所覆盖。
- 便利构造器必须先代理调用同一类中的其它构造器,然后再为任意属性赋新值。否则,便利构造器赋予的新值将被同一类中其它指定构造器所覆盖。
- 构造器在第一阶段构造完成之前,不能调用任何实例方法,不能读取任何实例属性的值,不能引用self
作为一个值,因为此时还没构造有效实例。
4. 总结两段式构造的详细流程
第一阶段:
- 某个指定构造器或便利构造器被调用。
- 完成新实例内存的分配,但此时内存还没有被初始化。
- 指定构造器确保其所在类引入的所有存储型属性都已赋初值。存储型属性所属的内存完成初始化。
- 指定构造器将调用父类的构造器,完成父类属性的初始化。这个调用父类构造器的过程沿着构造器链一直往上执行,直到到达构造器链的最顶部。
- 当到达了构造器链最顶部,且已确保所有实例包含的存储型属性都已经赋值,这个实例的内存被认为已经完全初始化。此时阶段 1 完成。
第二阶段:
从顶部构造器链一直往下,每个构造器链中类的指定构造器都有机会进一步定制实例。构造器此时可以访问self、修改它的属性并调用实例方法等等。
最终,任意构造器链中的便利构造器可以有机会定制实例和使用self
。
三、构造器的继承与重写
-
与OC不同,Swift中的子类默认情况不会自动继承父类的构造器,这是因为子类可能有更多新增属性,直接调用父类的构造器,可能会有一些属性无法初始化为出现错误。
-
如果我们希望子类拥有一个或多个与父类相同的构造器,那么这相当于子类重写了父类的构造器,需要在重写方法前添加override修饰符。(override的作用是提示编译器去检查父类中是否有相匹配的指定构造器,并验证构造器参数是否正确)
注意1: 重写系统自带的默认构造器(虽然不是显式的),也要带上override修饰符
注意2: 重写父类指定构造器必须带上override,即使你的子类将父类的指定构造器重写成了便利构造器
注意3: 子类编写和父类便利构造器相匹配的子类构造器时,由于子类不能直接调用父类的便利构造器,并不能看做是对父类构造器的重写。所以子类中“重写”父类便利构造器时,不需要加override修饰符。
四、构造器的自动继承
子类在默认情况下不会继承父类的构造器。但是如果满足特定条件,父类构造器还是是可以被自动继承的。
构造器的自动继承需要满足前提条件是:子类中引入的所有新属性都提供了默认值。
然后可分为两种情况:
情况1:子类没有定义任何指定构造器,它将自动继承所有父类的指定构造器。
class SuperClass {
var valueOne: Int
//指定构造器
init(valueOne: Int){
self.valueOne = valueOne
}
}
class SubClass: SuperClass {
var valueTwo:Int = 100 //子类的新引入属性
}
//自动继承了父类的构造器,这里被使用创建子类,子类新属性使用默认值
let subClass = SubClass(valueOne: 10)
print("subClass:(\(subClass.valueOne)) valueTwo(\(subClass.valueTwo))")
情况2:子类提供了所有父类指定构造器的实现(通过情况1继承过来的或者子类自定义实现),此时子类将自动继承所有父类的便利构造器。
class Animal{
var name:String
var age: Int
init(name:String, age:Int){
self.name = name
self.age = age
}
//父类的便利构造器
convenience init(name:String){
self.init(name:name, age:10)
}
}
class Person: Animal{
var nickName:String!
//实现了父类中的指定构造器
override init(name:String, age:Int){
super.init(name:name, age:age)
self.name = name
self.age = age + 2
}
}
let ps = Person(name: "zs", age: 18)
自动继承注意:
1.即使你在子类中添加了更多的便利构造器,这两条规则仍然适用。
2.对于情况 2,子类可以将父类的指定构造器实现为便利构造器。