一、构造函数的调用规则
swift规定了三条规则来限制构造函数之间的调用。
1、指定构造函数必须最终调用其直接父类的指定构造函数
2、便利构造函数必须调用当前类中定义的其他构造函数
3、便利构造函数必须最终导致一个指定构造函数被使用
class Person{
var name:String //这里不用加?是因为构造函数里边赋值了,不赋值就要加上?
var age:Int
//指定构造函数
init(age:Int,name:String) {
self.name = name
self.age = age
}
//便利构造函数
convenience init(age:Int){
self.init(age: age,name: "张三")
}
}
class Teacher : Person{
var sex:String
//指定构造函数
init(age:Int,name:String,sex:String){
self.sex = sex
super.init(age: age, name: name)
}
//便利构造函数
convenience init(age:Int){
self.init(age: age, name: "王五", sex: "老头")
}
}
var per = Person(age: 13)
print(per.name)
var tea:Teacher = Teacher(age:15,name:"李四",sex:"男")
print(tea.sex)
var tea2:Teacher = Teacher(age:38)
print(tea2.sex)
控制台
张三
男
老头
二、构造过程的安全检查
swift中类的构造过程分为两个阶段即调用构造方法分为两个阶段
- 1、首先分配内存,初始化子类新增的存储属性,然后沿构造函数链晚上初始化每个父类的存储属性,到达函数链的顶端。此时,子类和父类的所有存储属性都有初始值。
- 2、从顶部的构造链往下,给每个类一次机会在新的实例准备使用之前修改存储属性,挑用实例方法等。
安全检查一、
指定构造函数必须保证它所在类引入的所有存储属性确定之后才能完全初始化,之后才能将其他构造任务向上代理给父类中的构造函数。
这就是为什么self.sex = sex
要写在 super.init(age: age, name: name)
上边。
class Teacher : Person{
var sex:String
//指定构造函数
init(age:Int,name:String,sex:String){
self.sex = sex
super.init(age: age, name: name)
}
}
安全检查二、
指定构造函数必须先向上代理调用父类的构造函数,然后再为继承的属性设置新值。否则设置的新值将被父类的构造函数所覆盖
class Teacher : Person{
var sex:String
//指定构造函数
init(age:Int,name:String,sex:String){
self.sex = sex
self.name = "zhang"//这个name将被父类的赋值操作覆盖
super.init(age: age, name: name)
}
}
安全检查三、
便利构造函数必须先调用同一类中的其他构造函数,然后再为任意属性赋值新值,否则新值将被本类中的其他构造函数覆盖
class Teacher : Person{
var sex:String
//指定构造函数
init(age:Int,name:String,sex:String){
self.sex = sex
super.init(age: age, name: name)
}
//便利构造函数
convenience init(age:Int){
self.sex = "zhang"//这个name将被指定构造函数的赋值操作覆盖
self.init(age: age, name: "王五", sex: "老头")
}
}
//由于我看的书是swift2.0,swift更新到3.0之后这样写会直接报错,但是至少原理是酱紫的不是。万变不离其宗,报错信息如下
//Use of 'self' in property access 'sex' before self.init initializes self
安全检查四、
构造函数再第一步分配内存完成之前,不能调用任何实例方法,不能读取任何实例属性的值
class Teacher : Person{
var sex:String
//指定构造函数
init(age:Int,name:String,sex:String){
self.sex = sex
self.description() //error
super.init(age: age, name: name)
}
func description(){
print("description")
}
}
三、构造函数的自动继承
子类对父类构造函数的自动继承有两个规则
1、如果子类中定义的所有新属性都有默认值,并且子类没有自定义任何构造函数,那么子类将自动继承父类的所有指定构造函数
2、如果子类提供了所有父类指定构造函数的实现,无论是通过规则1继承的,还是提供了自定义实现,它将自动继承所有父类额便利构造函数(即使属性没有默认值,只要实现了父类的所有指定构造函数,就会自动继承父类的所有便利构造函数)。而且,子类可以将父类的指定构造函数实现为便利构造函数
//个人感觉这个东西没什么用,你都定义了子类了,肯定要增加属性或者方法,还不自己整自己的构造函数,还用父类的,丢人不丢人,哈哈
四、构造函数的重写
重写父类构造函数,跟重写方法一样。override即可