Swift构造器重载


与函数一样,方法也存在重载,其重载的方式与函数一致。那么作为构造器的特殊方法,是否也存在重载呢?答案是肯定的。
一、构造器重载概念
Swift中函数重载的条件也适用于构造器,条件如下:
函数有相同的名字;
参数列表不同或返回值类型不同,或外部参数名不同;
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) ⑤ println("长方形:/(rectc1.width) x /(rectc1.height)")   var rectc2 = Rectangle(W : 320.0, H : 480.0) ⑥ println("长方形:/(rectc2.width) x /(rectc2.height)")   var rectc3 = Rectangle(length: 500.0) ⑦ println("长方形3:/(rectc3.width) x /(rectc3.height)")   var rectc4 = Rectangle() ⑧ println("长方形4:/(rectc4.width) x /(rectc4.height)")

上述代码第①~④行定义了4个构造器,其他是重载关系。从参数个数和参数类型上看,第①行和第②行的构造器是一样的,但是它们的外部参数名不同,所以在第⑤行调用的是第①行的构造器,第⑥行调用的是第②行的构造器。
第③行和第④行的构造器参数个数与第①行不同,所以在第⑦行调用的是第③行的构造器,第④行调用的是第⑧行的构造器。
二、值类型构造器代理
为了减少多个构造器间的代码重复,在定义构造器时,可以通过调用其他构造器来完成实例的部分构造过程,这个过程称为构造器代理。构造器代理在值类型和引用类型中使用方式不同,本节我们先介绍值类型构造器代理。
将上一节的示例修改如下:

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(W : length, H : length)     }       init() {      ④         self.init(width: 640.0, height: 940.0)     }   }   var rectc1 = Rectangle(width : 320.0, height : 480.0) ⑤ println("长方形:/(rectc1.width) x /(rectc1.height)")   var rectc2 = Rectangle(W : 320.0, H : 480.0) ⑥ println("长方形:/(rectc2.width) x /(rectc2.height)")   var rectc3 = Rectangle(length: 500.0) ⑦ println("长方形3:/(rectc3.width) x /(rectc3.height)")   var rectc4 = Rectangle() ⑧ println("长方形4:/(rectc4.width) x /(rectc4.height)") 

将Rectangle声明为结构体类型,其中也有4个构造器重载。在第③行和第④行的构造器中使用了self.init语句,self指示当前实例本身,init是本身的构造器,第③行的self.init(W : length, H : length)语句是在调用第②行定义的构造器,第④行的self.init(width: 640.0, height: 940.0)语句是在调用第①行定义的构造器。
这种在同一个类型中通过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) ⑤ println("长方形:/(rectc1.width) x /(rectc1.height)")   var rectc2 = Rectangle(W : 320.0, H : 480.0) ⑥ println("长方形:/(rectc2.width) x /(rectc2.height)")   var rectc3 = Rectangle(length: 500.0) ⑦ println("长方形3:/(rectc3.width) x /(rectc3.height)")   var rectc4 = Rectangle() ⑧ println("长方形4:/(rectc4.width) x /(rectc4.height)") 

将Rectangle声明为类,其中也有4个构造器重载。在第③行和第④行的构造器中使用了self.init语句,并且在构造器前面加上了convenience关键字,convenience表示便利构造器,这说明我们定义构造器是横向代理调用其他构造器。

第③行的self.init(W : length, H : length)语句是在横向调用第②行定义的构造器代理,第④行的self.init(width: 640.0, height: 940.0)语句是在横向调用第①行定义的构造器代理。



更多内容请关注国内第一本Swift图书《Swift开发指南》
本书交流讨论网站:http://www.51work6.com/swift.php
欢迎加入Swift技术讨论群:362298485

欢迎关注智捷iOS课堂微信公共平台

Swift构造器重载


发表评论

电子邮件地址不会被公开。

您可以使用这些HTML标签和属性: <a href="" title=""> <abbr title=""> <acronym title=""> <b> <blockquote cite=""> <cite> <code> <del datetime=""> <em> <i> <q cite=""> <strike> <strong>