构造函数的主要作用是初始化实例,其中包括:初始化存储属性和其它的初始化。在Rectangle类或结构体中,如果在构造函数中初始化存储属性width和height后,那么在定义他们时就不需要初始化了。
Rectangle类代码如下:
- class Rectangle {
- var width: Double
- var height: Double
- init() {
- width = 0.0
- height = 0.0
- }
- }
如果存储属性在构造函数中没有初始化,在定义的时候也没有初始化,那么就会发生编译错误。
构造函数还可以初始化常量存储属性,下面我们看示例代码:
- class Employee {
- let no: Int //常量类型属性。曾讲过常量只能在定义的同时赋值,而在构造函数中,常量属性可以不遵守这个规则
- var name: String? //存储属性一般在定义的时候初始化。如果不能确定初始值,可以采用可选类型属性
- var job: String? //存储属性
- var salary: Double
- var dept: Department? //存储属性
- init() {
- no = 0 //在构造函数中赋值,这种赋值不能放在普通方法中
- salary = 0.0
- dept = nil
- }
- }
- struct Department {
- let no: Int //常量类型属性
- var name: String
- init() {
- no = 10 //在构造函数中赋值,这种赋值不能放在普通方法中
- name = "SALES"
- }
- }
- let dept = Department()
- var emp = Employee()
使用外部参数名
为了增强程序的可读性,Swift中的方法和函数可以使用外部参数名。在构造函数中也可以使用外部参数名。构造函数中的外部参数名要比一般的方法和函数更有意义,由于构造函数命名都是init,如果一个类型中有多个构造函数,我们可以通过不同的外部参数名区分调用不同的构造函数。
下面看示例代码:
- class RectangleA {
- var width: Double
- var height: Double
- init(W width: Double,H height: Double) { //定义构造函数有两个参数width和height,并且为参数提供了外部参数名W和H。
- self.width = width //函数参数赋值给属性
- self.height = height //
- }
- }
- var recta = RectangleA(W: 320, H: 480) //创建RectangleA实例,这里使用了外部参数名
- print("长方形A:\(recta.width) x \(recta.height)")
这里我们定义的是类,但也完全适用于结构体。
构造函数中的局部参数名可以直接作为外部参数名使用。
下面看示例代码:
- class RectangleB {
- var width: Double
- var height: Double
- init(width: Double, height: Double) { //构造函数,其中没有声明外部参数名。
- self.width = width
- self.height = height
- }
- }
- var rectb = RectangleB(width: 320, height: 480) //代码调用构造函数时,我们使用了外部参数名width和height,这些外部参数名就是局部参数名
- print("长方形B:\(rectb.width) x \(rectb.height)")
构造函数作为一种特殊方法,也可以重载。
Swift中构造函数可以多个,他们参数列表和返回值可以不同,这些构造函数构成重载。
示例代码如下:
- class Rectangle {
- var width: Double
- var height: Double
- init(width: Double, height: Double) {
- self.width = width
- self.height = height
- }
- init(W width: Double,H height: Double) {
- self.width = width
- self.height = height
- }
- init(length: Double) {
- self.width = length
- self.height = length
- }
- init() {
- self.width = 640.0
- self.height = 940.0
- }
- }
- var rectc1 = Rectangle(width: 320.0, height: 480.0)
- print("长方形:\(rectc1.width) x \(rectc1.height)")
- var rectc2 = Rectangle(W: 320.0, H: 480.0)
- print("长方形:\(rectc2.width) x \(rectc2.height)")
- var rectc3 = Rectangle(length: 500.0)
- print("长方形3:\(rectc3.width) x \(rectc3.height)")
- var rectc4 = Rectangle()
- print("长方形4:\(rectc4.width) x \(rectc4.height)")
构造函数代理
为了减少多个构造函数间的代码重复,在定义构造函数时,可以通过调用其他构造函数来完成实例的部分构造过程,这个过程称为构造函数代理。构造函数代理在结构体和类中使用方式是不同,先介绍结构体中构造函数代理。
将上一节的示例修改如下:
- struct Rectangle {
- var width: Double
- var height: Double
- init(width: Double, height: Double) {
- self.width = width
- self.height = height
- }
- init(W width: Double,H height: Double) {
- self.width = width
- self.height = height
- }
- init(length: Double) { //调用了self.init语句
- self.init(W: length, H: length)
- }
- init() { //调用了self.init语句
- self.init(width: 640.0, height: 940.0)
- }
- }
- var rectc1 = Rectangle(width: 320.0, height: 480.0)
- print("长方形:\(rectc1.width) x \(rectc1.height)")
- var rectc2 = Rectangle(W: 320.0, H: 480.0)
- print("长方形:\(rectc2.width) x \(rectc2.height)")
- var rectc3 = Rectangle(length: 500.0)
- print("长方形3:\(rectc3.width) x \(rectc3.height)")
- var rectc4 = Rectangle()
- print("长方形4:\(rectc4.width) x \(rectc4.height)")
将Rectangle声明为结构体类型,其中也有4个构造函数重载。
这种在同一个类型中通过self.init语句进行调用当前类型其它构造函数,其它构造函数被称为构造函数代理。
类构造函数横向代理
由于类有继承关系,类构造函数代理比较复杂,分为横向代理和向上代理。
横向代理类似于结构体类型构造函数代理,发生在同一类内部,这种构造函数称为便利构造函数(convenience initializers)。
向上代理发生在继承情况下,在子类构造过程中要先调用父类构造函数,初始化父类的存储属性,这种构造函数称为指定构造函数(designated initializers)。
将上面的示例修改如下:
- class Rectangle {
- var width: Double
- var height: Double
- init(width: Double, height: Double) {
- self.width = width
- self.height = height
- }
- init(W width: Double,H height: Double) {
- self.width = width
- self.height = height
- }
- convenience init(length: Double) {
- self.init(W: length, H: length)
- }
- convenience init() {
- self.init(width: 640.0, height: 940.0)
- }
- }
- var rectc1 = Rectangle(width: 320.0, height: 480.0)
- print("长方形:\(rectc1.width) x \(rectc1.height)")
- var rectc2 = Rectangle(W: 320.0, H: 480.0)
- print("长方形:\(rectc2.width) x \(rectc2.height)")
- var rectc3 = Rectangle(length: 500.0)
- print("长方形3:\(rectc3.width) x \(rectc3.height)")
- var rectc4 = Rectangle()
- print("长方形4:\(rectc4.width) x \(rectc4.height)")
将Rectangle声明为类,其中也有4个构造函数重载。