本章包含的内容的如下:
构造过程是使用类,结构体或枚举型的实例之前的准备过程。在新实例可用钳必须执行这个过程,具体操作包括设置实例中的每个存储属性的初始值和执行其他必须的设置或初始化工作
通过定义构造器(Initializers)来实现构造过程,这些构造器可以看做是用来创建特定类型新实例的特殊方法。与Objc中的构造器不同,Swift的构造器无需返回值,它们的主要任务是保证新实例在第一次使用前完成正确的初始化。
类的实例也可以通过定义析构器(deinitializer)在实例释放前执行特定的清除工作。
存储属性的初始赋值
类和结构体在创建实例时,必须为所有的存储属性设置合适的初始值。存储型属性的值不能处于一个未知的状态。
你可以在构造器中为存储属性赋初值,也可以在定义属性时为其设置默认值。以下小节详细讨论这两个方法。
注意 当你为存储型属性设置默认值或者在构造器中为期赋值时,它们的值是被直接设置的,不会触发任何属性观察期(property observers)
构造器
构造器在创建某个特定类型的新实例时被调用。它的最简单形式类似于一个不带任何参数的实例方法,以关键字init命名:
1 2 3 |
|
下面例子中定义了一个用来保存华氏温度的结构体Fahrenheit,它拥有一个double
类型的存储型属性 temperature:
1 2 3 4 5 6 7 8 9 10 |
|
这个结构体定义了一个不带参数的构造器init,并在里面将存储属性temperature 的值初始化为32
默认初始值
如前所述,你可以在构造器中为存储型属性设置初始化,同样,你也可以在属性声明时为其设置默认值
注意 如果一个属性总是使用相同的初始值,那么为其设置一个默认值比每次都在构造器中赋值要好。两种方法的效果是一样的。只不过使用默认值让属性的初始化和声明结合的更紧密。使用默认值能让你的构造器更简洁,更清晰,且能通过默认值自动推导出属性的类型;同时,它也能让你充分利用默认构造器,构造器继承等特性
你可以使用更简单的方式在定义结构体Fahrenheit时为属性temperature设置默认值:
1 2 3 |
|
自定义构造过程
你可以通过输入参数可可选类型的属性来自定义构造过程,也可以在构造过程中修改常量属性。
构造参数
自定义构造过程时,可以在定义中提供构造函数,指定所需值的类型和名字。构造参数的功能和语法跟函数的方法的参数相同。
下面例子中定义了一个包含摄氏温度的结构体Celsius
.它定义了两个不同的构造器:init(fromFahrenheit:)和init(fromKelvin:),两者分别通过接受不同温标下的温度值来创建新的实例:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
第一个构造器拥有一个构造参数,其外部名字为fromFahrenheit,内部名字为fahrenheit;第二个构造器也拥有一个构造函数,其外部名字为 fromKelvin,内部名字为 kelvin.这两个构造器都将唯一的参数值转换成摄氏温度值,并保存在属性temperatureInCelsius中
参数的内部名称和外部名称
跟函数和方法参数相同,构造参数也拥有一个在构造器内部使用的参数名字和一个在调用构造器时使用的外部参数名字。
然而,构造器并不像函数和方法那样在括号前有一个可辨别的名字。因此在调用构造器时,主要工作构造器中的参数名和类型来确定应该被调用的构造器。正因为参数如此重要,如果你在定义构造器时没有提供参数的外部名字,Swift会为构造器的每个参数自动生成一个跟内部名字相同的外部名.
以下例子中定义了一个结构体Color,它包含了三个常量:red
,green
和blue
.这些属性可以存储在0-1之间的值,用来表示红绿蓝成分的含量
Color
提供了一个构造器,其中包含三个Double
类型的构造函数,Color也可以提供第二个狗奥奇,它只包含名为white的double类型的参数,它被用于给上述三个构造参数赋予同样的值。
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
两种构造器都能用于创建一个新的color实例,你需要为构造器每个外部参数传入值:
1 2 |
|
注意,如果不通过外部参数名子传值,你是没法调用这个构造器的,只要构造器定义了某个外部参数名,你就必须使用它,忽略它将导致编译错误:
1 2 |
|
不带外部名的构造参数
如果你不希望为构造器的某个参数提供外部名字,你可以使用下划线(_)来显式描述它的外部名,以此重写上面所说的默认行为.
下面是之前的Celsius例子的扩展,跟之前相比添加了一个带有double类型参数的构造器,其外部名用_代替:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|
调用Celsius(37.0)意图明确,不需要外部参数名称。以此适合使用
Celsius(37.0)
这样的构造器,从而可以通过提供Double类型的参数值调用构造器,而不需要加上外部名。
可选属性类型
如果你你定制的类型包含一个逻辑上允许取值为空的存储性属性 – 无论是因为它无法再初始化时赋值,还是因为它在之后某个时间点可以赋值为空–你都需要将它定为可选类型(optional type).可选类型的属性将自动初始化为nil,表示这个属性是有意在初始化时设置为空的。
下面的例子中定义了类SurveyQuestion
,它包含了一个可选字符串属性response:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|
迪奥哈问题的答案在回答前是无法确定的,因此我们将属性response声明为string?类型,或者说是可选字符串类型,当SurveyQuestion
实例化时,它将自动赋值为nil,表明此字符串暂时还没有值
构造过程中的属性修改
你可以在构造过程中的任意时间点给常量属性指定一个值,只要在构造过程结束时是一个确定的值。一旦常量尚需经被赋值,它将永远不可更改。
注意: 对于类的实例来说,它的常量属性只能在定义它的类的构造过程中修改;不能在子类中修改
你可以修改上面的SurveyQuestion实例,用常量属性替代变量属性text,表示问题内容text在SurveyQuestion的实例被创建之后不会再被修改。尽管text属性现在是常量,我们仍然可以在类的构造器中设置它的值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|
默认构造器
如果结构体或类的所有属性都有默认值,同事没有自定义的构造器,那么Swift会给这些结构体或类提供一个默认的构造器(default initalilizers)。这个默认构造器将简单地创建一个所有属性值都设置为默认值的实例。
下面例子中创建了一个类ShoppingListItem,它封装了购物清单中的某一个物品的属性:名字,数量和购买状态
1 2 3 4 5 6 |
|
由于ShoppingListItem类中所有属性都有默认值,且它没有父类的基类,它将自动获得一个可以为所有属性设置默认值的默认构造器,上面例子中使用默认构造创造了一个ShoppingListItem类的实例,并将其赋值给变量 item
结构起的逐一成员的构造器
除了上面提到的默认构造器,如果结构起没有提供自定义的构造器,他们将自动获得一个逐一成员构造器,及时结构天气的存储型尚需经没有默认值。
逐一成员构造器是用来初始化结构体新实例成员属性的快捷方法。我们在调用逐一成员构造器时,通过与成员属性名相同的参数名进行传值来完成对成员属性的初始赋值
下面例子中定义了结构体size,包含两个属性width和height。swift可以根据这两个属性的初始赋值0自动推导出他们的类型为double
结构体size自动获得了一个逐一成员构造器init(width:height:)。你可以用它来为size创建新的实例:
1 2 3 4 |
|
值类型的构造器代理
构造器可以通过调用其它构造器来完成实例的部分构造过程。这一过程成为构造器代理,它能减少多个构造器间的代码重用。
构造器代理的实现规则和形式在值类型和类类型中有所不同。值类型(结构体和枚举类型)不支持继承,所以构造器代理的过程相对简单,因为它们只能代理给自己的其它构造器。类则不同,它可以继承自其它类,这意味着类有责任保证其所有继承的存储型属性在构造时也能正确的初始化。
对于值类型,你可以使用self.init在自定义的构造器中引用相同类型中的其它构造器。并且你只能在构造器内部调用self.init
如果你为某个值类型定义了一个自定义的构造器,你将无法访问到默认构造器(如果是结构体,还将无法访问逐一成员的构造器)。这种限制可以防止你为值类型增加了一个额外的且十分负载的构造器之后,仍然有人错误的使用自动生成的构造器
注意 如果你希望默认构造器,逐一成员构造器以及你自己的自定义的构造器都能用来创建实例,可以将自定义的构造器写到扩展中,而不是写在值类型的原始定义中。
下面的例子将定义一个结构体 rect,用来代表几何矩形。这个例子需要两个辅助的结构体Size和point,它们各自为其所有的属性提供了默认值0
1 2 3 4 5 6 |
|
你可以通过以下三种方式为Rect创建实例–使用被初始化为默认值的origin和Size属性来初始化;提供指定的origin和size实例来初始化;提供指定的center和size来初始化。在下面Rect结构体定义中,我们为这三种方式提供了三个自定义的构造器:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
|
第一个Rect构造器init(),在功能上跟没有自定义构造器自动获得的默认构造器是一样的。这个构造器是一个空函数,使用一对大括号{}来表示,它没有指定任何构造过程。调用这个构造器将返回一个Rect实例,它的origin和size属性都是用定义时的默认值Point(x: 0.0, y: 0.0)和Size(width: 0.0, height: 0.0):
1 2 |
|
第二个Rect构造器init(origin:size:),在功能上跟结构体在没有自定义构造器时获得的逐一成员构造器是一样的。这个构造器只是简单的将origin和size的参数值赋给对应的存储性属性:
1 2 3 |
|
第三个Rect构造器init(center:size:)稍微复杂一点。它先通过center和size的值计算出origin的坐标,然后再调用init(origin:size:)构造器来将新的origin和size值赋值到对应的属性中:
1 2 3 |
|
构造器init(center:size:)可以直接将origin和size 的新值赋值到对应的属性中。然而,利用恰好提供了相关功能的现有构造器会更为方便,构造器init(center:size:)的意图也会更加清晰
类的集成和构造过程
类里面的所有存储性属性–包括所有继承自父类的属性–都必须在构造过程中设置初始值
Swift为类类型提供了两种构造器来确保实例中所有存储性属性都能获得初始值,它们分别是指定构造器和便利构造器
指定构造器和便利构造器
指定构造器是类中最主要的构造器。一个指定构造器将初始化类中提供的所有属性,并根据父类链上调用父类的构造器来实现父类的初始化
每一个类都必须拥有至少一个指定构造器。在某些情况下,许多类通过继承了父类中的指定构造器而满足了这个条件。
便利构造器(convenience initializers)是类中比较次要的,辅助性的构造器。你可以定义便利构造器来调用同一个类中的指定构造器,并为其参数提供默认值。你也可以定义便利构造器来创建一个特殊用途或者特定输入值的实例
你应当只在必要的时候为类提供便利构造器,比方或某种情况下通过使用便利构造器来快捷调用某个指定构造器,能够节省更多开发时间并让类的构造郭恒更清晰明了
指定构造器和便利构造器的语法
类的指定构造器的写法跟值类型简单构造器一样:
1 2 3 |
|
便利构造器也采用相同样式的写法,但需要在init关键字之前放置convenience关键字,并使用空格将它们俩分开:
1 2 3 |
|
类的构造器代理规则
为了简化指定构造器和便利构造器之间的调用关系,Swift采用以下三条规则来限制构造器之间的代理调用:
- 规则1 指定构造器必须调用其直接父类的指定构造器
- 规则2 便利构造器必须调用同一类中定义的其它构造器
- 规则3 便利构造器必须最终导致一个指定构造器被调用
一个更方便记忆的方式是:
- 指定构造器必须总是向上代理
- 便利构造器必须总是横向代理
这些规则可以通过下面的图例来说明:
如图所示,父类中包含一个指定构造器和两个便利构造器。其中一个便利构造器调用了另外一个便利构造器,而后者又调用了唯一的指定构造器。这满足了上面提到 规则2和3.而两个指定构造器必须调用父类中唯一的指定构造器,这满足了规则1
注意 这些规则不会影响类的实例如何创建。任何上图中展示的构造器都可以用来创建完全初始化的实例。这些规则只影响类定义如何实现。
下图中展示了一种涉及四个类的更复杂的类层次结构。它演示了指定构造器是如何在类层次中充当“管道”的作用,在类的构造器链上简化了类之间的相互关系。
两段式构造过程
Swift中类的构造过程包含两个阶段。第一个阶段,每个存储属性被引入它们的类指定一个初始值。当每个存储性属性的初始值被确定后,第二个阶段开始,它给每个类一次机会,在新实例准备使用之前进一步定制它们的存储性属性。
两段式构造过程的使用让构造过程更安全,同时在整个类层级结构中给予了每个类完全的灵活性。两段式构造过程可以防止属性值在初始化之前被访问,也可以防止属性被另外一个构造器意外地赋予不同的值。
注意 Swift的两段时构造过程跟Objective-C中的构造过程类似。最主要的区别在于阶段1,Objective-C给每一个属性赋值0或空值(比如说0或nil)。Swift的构造流程则更加灵活,它允许你设置定制的初始值,并自如应对某些属性不能以0或nil作为合法默认值的情况。
Swift编译器将执行4种有效的安全检查,已确保两段式构造过程能不出错地完成:
安全检查1
指定构造器必须保证它所在类引入的所有属性都必须先初始化完成,之后才能将其它构造任务向上代理给父类的构造器
如上所述,一个对象的内存只有在其所有存储性苏醒确定之后次啊能完全初始化。为了满足这一规则,指定构造器必须保证它所在类引入的属性在它往上代理之前先完成初始化。
安全检查2
指定构造器必须先向上代理调用父类构造器,然后再为继承的属性设置新值。如果没这么做,指定构造器赋予的新值将被父类中的构造器所覆盖。
安全检查3
便利构造器必须先代理调用同一类中的其它构造器,然后再为任意属性赋新值。如果没这么做,便利构造器赋予的新值将被同一类中的其它指定构造器所覆盖
安全检查4
构造器在第一阶段构造完成之前,不能调用任何实例方法,不能读取任何实例属性的值,不能引用self作为一个值。
类实例在第一阶段结束以前并不是完全有效的。只有第一阶段完成后,该实例会成为有效实例,才能访问属性和调用方法。
以下是两段式构造过程中基于上述安全检查的构造流程展示:
阶段1
- 某个指定构造器或便利构造器被调用
- 完成新实例内存的分配,但此时内存还没有被初始化
- 指定构造器确保其所在类引入的所有存储型属性都已赋初值。存储型属性所属的内存完成初始化
- 指定构造器将调用父类的构造器,完成父类属性的初始化
- 这个调用父类的构造器的过程沿着构造器链一直往上执行,知道到达构造器链的最顶层
- 当到达了构造器链的最顶部,且已确保所有势力包含的存储型属性都已经赋值,这个实例的内存被认为已经完成初始化。此时阶段1完成
阶段2
- 从顶部构造器链一直往下,每个构造器链中类的指定构造器都有机会进一步定制实例。构造器此时可以访问self.修改它的属性并调用实例方法等
- 最终,任意构造器链中的便利构造器可以有机会定制实例和使用self
下图展示了在假定的子类和父类之间的构造阶段1
在这个例子中,构造过程从对子类中一个便利构造器的调用开始。这个便利构造器此时没法修改任何属性,它把构造任务代理给同一类中的指定构造器。
如安全检查1所示,指定构造器将确保所有子类的属性都有值。然后它将调用父类的指定构造器,并沿着构造器链一直往上完成父类的构造过程
父类中的指定构造器确保所有父类的属性都有值。由于没有更多的父类需要初始化,也就无需继续向上代理。
一旦父类中所有属性都有了初始值,实例的内存被认为是完成初始化,阶段1完成。
以下展示了相同构造过程的阶段2:
父类中的指定构造器现在有机会进一步来定制实例。
一旦父类中的指定构造器完成调用,子类中的指定构造器可以执行更多的定制操作。 最终,一旦子类的指定构造器完成调用,最开始被调用的便利构造器可以执行更多的定制操作
构造器的继承和重写
跟Objc中的子类不同,Swift中的子类默认情况下不会继承父类的构造器。Swift的这种机制可以防止一个父类的简单构造器被一个更精细的子类继承,并被错误地用来创建子类的实例。
注意
父类的构造器仅会在安全和适当的情况下被继承
假如你希望自定义的子类中能提供一个或多个父类相同的构造器,你可以在子类中提供这些构造器的自定义实现。
当你编写与一个和父类中指定构造器相匹配的子类构造器时,你实际上是在重写父类的这个指定构造器。因此,你必须在定义子类构造器时带上 override
修饰符。即使你重写的是系统自动提供的默认构造器,也需要带上override
修饰符。
正如重写属性,方法或者是下标,override修饰符会让编译器去检查父类中是否有相匹配的指定构造器,并验证构造器参数是否正确。
注意 当你重写了父类的指定构造器时,你总是需要写override修饰符,即使你的子类的指定构造器重写为了便利构造器
相反,如果你编写了一个和父类便利构造器相匹配的子类构造器,由于子类不能直接调用父类的便利构造器,因此,严格意义上来讲,你的子类并未对一个父类构造器提供重写。最后的结果就是,你在子类中“重写”一个父类便利构造器时,不需要加override前缀。
在下面的例子中定义了一个叫做Vehicle
的基类。基类中声明了一个存储型属性numberOfWheels
,它是值为0的Int类型的存储型属性。numberOfWheel属性用于创建名为descrpiption的String
类型的计算型属性:
1 2 3 4 5 6 |
|
Vehicle
类只为存储型属性提供默认值,而不自定义构造器。因此,它会自动获得一个默认构造器,自动获得的默认构造器总会是类中的指定构造器,它可以用于创建numberOfWheels为0的Vehicle实例:
1 2 3 |
|
下面例子中,定义了Vehicle的子类Bicycle:
1 2 3 4 5 6 |
|
子类Bicycle
定义了一个自定义的指定构造器init(),这个指定构造器和父类的指定构造器匹配,所以Bicycle
中的指定构造器需要带上override修饰符
Bicycle
的构造器init()以调用super.init()方法开始,这个方法的作用是调用Bicycle的父类Vechicle
的默认构造器,这样可以确保Bicycle在修改属性之前,它所继承的属性numberOfWheels能被Vehicle
类初始化。在调用super.init()之后,属性numberOfWheels
的原值被新值2替换。
如果你创建了Bicycle
实例,你可以调用继承的description
计算型属性去查看属性numberOfWheels
是否有改变:
1 2 3 |
|
注意
子类可以在初始化时修改继承来的变量属性,但是不能修改继承来的常量属性
构造器的自动继承
如上所述,子类在默认情况下不会继承父类的构造器。但是如果满足特定条件,父类构造器是可以被自动继承的。在实践中,这意味着对于许多常见场景你不必重写父类的构造器,并且可以在安全的情况下以最小的代价继承父类的构造器。
假如你为子类中引入的所有新属性都提供了默认值,以下2个规则适用:
规则1
如果子类没有定义任何指定构造器,它将自动继承所有父类的指定构造器
规则2
如果子类提供了所有父类指定构造器的实现–无论是通过规则1继承过来的,还是提供了自定义实现–它将自动继承所有父类的便利构造器。
即使你在子类中添加了更多的便利构造器,这两条规则仍然适用
注意 对于规则2,子类可以将父类的指定构造器实现为便利构造器
总结:
如果子类继承父类,在子类的构造器中,如果是继承的属性,则先调用
super.init(...)
方法,先让父类完成属性初始化,然后再对继承的属性赋值;如果子类中独有的属性,则先完成子类属性的初始化,再调用父类的
super.init(...)
方法
指定构造器和便利构造器实践
接下来的例子将在实践中展示知道你给构造器,便利构造器以及构造器的自动继承。这个例子定义了包含三个类Food
,RecipeIngredient
以及ShoppingListItem
的类层次结构,并将演示他们的构造器是如何相互作用的。
类层次中的基类是Food,它是一个简单的用来封装食物名字的类。Food类引入了一个叫做name的String类型的属性,并且提供了两个构造器来创建Food
实例:
1 2 3 4 5 6 7 8 9 10 |
|
下图中展示了Food
的构造器链:
类类型没有默认的逐一成员构造器,所以Food
类提供了一个接受单一参数name的指定构造器。这个构造器可以使用一个特定的名字来创建新的Food
实例:
1 2 |
|
Food
类中的构造器init(name: String)
被定义为一个指定构造器,因为它能确保Food
实例的所有存储型属性都被初始化。Food
类没有父类,所以init(name: String)
构造器不需要调用super.init()
来完成构造过程。
Food类同样提供了一个没有参数的便利构造器init().这个init()构造器为新食物提供了一个默认的占位名字,通过横向代理指定构造器init(name: String)
并给参数name传值[unnamed]来实现:
1 2 |
|
类层级中的第二类是Food的子类RecipeIngredient
。RecipeIngredient
类构建了食谱中的一味调味剂。它引入了Int类型的属性quantity
(以及从Food继承过来的name属性),并且定义了两个构造器来创建RecipeIngredient
实例:
1 2 3 4 5 6 7 8 9 10 11 |
|
下图中展示了RecipeIngredient
类的构造器链:
RecipeIngredient
类拥有一个指定构造器init(name: String, quantity: Int)
,它可以用来填充RecipeIngredient
实例的所有属性值。这个构造器一开始先将传入的quanity
参数赋值给quantity
属性,这个属性也是唯一在RecipeIngredient
中新引入的尚需经。随后,构造器向上代理到父类Food
的init(name: String)
。这个过程满足 两段式构造过程中 的安全检查1
RecipeIngredient
还定义了一个便利构造器 init(name: String)
,它只通过 name
来创建RecipeIngredient
的实例。这个便利构造器假设任意RecipeIngredient
实例的quantity
为1,所以不需要显式指明数量即可创建出实例。
这个便利构造器的定义可以更加方便和快捷地创建实例,并且避免了创建多个quantity
为1的RecipeIngredient
实例时的代码重复。这个便利构造器只是简单地横向代理到类中的指定构造器,并且为quantity参数传递1.
注意,RecipeIngredient
的便利构造器init(name: String)
使用了跟Food
中指定构造器init(name: String)
相同的参数。由于这个便利构造器重写了父类的指定构造器init(name: String)
,因此必须在前面使用override修饰符
尽管RecipeIngredient
将父类的指定构造器重写为了便利构造器,它依然提供了父类的所有指定构造器的实现。因此,RecipeIngredient
会自动继承父类的所有便利构造器。
在这个例子中,RecipeIngredient
的父类是Food
,它有一个便利构造器init().这个便利构造器会被RecipeIngredient
继承。这个继承版本的init()在功能上跟Food提供的版本是一样的,只是它会代理到RecipeIngredient
版本的init(name: String)
而不是Food提供的版本。
所有的这三种构造器都可以用来创建新的RecipeIngredient
实例:
1 2 3 |
|
类层级中的第三个也是最后一个类是RecipeIngredient
的子类,叫做ShoppingListItem
,这个类构建了购物单中出现的某一种调味料。
购物单中每一项总是从未购买状态开始的。为了呈现这一事实,ShoppingListItem
引入了布尔类型的属性purchased
,它的默认值是false
。ShoppingListItem
还添加了一个计算型属性description
,它提供了关于ShoppingListItem
实例的一个文字描述:
1 2 3 4 5 6 7 8 |
|
注意:
ShoppingListItem
没有定义构造器来为purchased
提供初始值,因为添加到购物单的物品的初始状态总是未购买。
由于它为自己引入的所有属性都提供了默认值,并且自己没有定义任何构造器,ShoppingListItem
将自动继承所有父类中的指定构造器和便利构造器。
下团展示了这三个类的构造器链:
你可以使用全部三个继承来的构造器来创建ShoppingListItem
的新实例:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
如上所述,例子中通过字面量方式创建了一个数组breakfastList
,它包含了三个
ShoppingListItem
实例,因此数组的类型也能被自动推导为[ShoppingListItem]
.在数组创建完之后,数组中第一个ShoppingListItem
实例的名字从[unnamed]更改为 orange juice
,并标记为已购买。打印数组中每个元素的描述显示了它们都已按照预期被赋值。
可失败构造器
如果一个类,结构起或枚举类型的对象,在构造过程中有可能失败,则为其定义个可失败的构造器。这里所指的失败
是指,如给构造器传入无效的参数值,或缺少某种所需的外部资源,又或者是不满足某种必要的条件等。
为了妥善处理这种构造过程中可能会失败的情况,你可以在一个类,结构体或是枚举类型的定义中,添加一个或多个可失败构造器。其语法为在init关键字后面添加问号(init?).
注意:
可失败构造器的参数名和参数类型,不能为其它非可失败构造器的参数名,及其参数类型相同
可失败构造器会创建一个类型为自身类型的可选类型的对象。你通过return nil
语句来表明可失败构造器在何种情况下应该失败
。
注意
严格来说,构造器都不支持返回值。以为构造器本身的租用,只是为了确保对象能被正确构造。因此你只是用
return nil
表明可失败构造器构造失败,而不要用关键字return来表明构造成功
下例中,定义了一个名为Animal
的结构体,其中有一个名为species
的String类型的常量属性。同时该结构体还定义了一个接受一个名为species
的String类型参数的可失败构造器。这个可失败构造器检查传入的参数是否为一个空字符串。如果为空字符串,则构造失败。否则,species
属性被赋值,构造成功。
1 2 3 4 5 6 7 |
|
你可以通过该可失败的构造器来创建一个Animal的实例,并检查构造过程是否成功。
1 2 3 4 5 6 7 |
|
如果你给改可失败构造器传入一个空字符串作为参数,则会导致构造失败:
1 2 3 4 5 6 7 |
|
注意
空字符串(如"“,而不是"Giraffe")和一个值为nil的可选类型的字符串是两个完全不同的概念。上例中的空字符串(”“)其实是一个幼小的,非可选类型的字符串。这里我们只所以让Animal的可失败构造器失败,只是因为对于
Animal
这个类的species
属性来说,它更适合有一个具体的值,而不是空字符串。
枚举类型的可失败构造器
你可以通过一个带一个或多个参数的可失败构造器来获取枚举类型中特定的枚举成员。如果提供的参数无法匹配任何枚举成员,则构造失败。
下例中,定义了一个名为TemperatureUnit
的枚举类型。其中包含了三个可能的枚举成员(Kelvin,Celsius,和Fahrenheit),以及一个根据Character
值找出所对应的枚举成员的可失败构造器:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
你可以利用该可失败构造器在三个枚举成员中获取一个相匹配的枚举成员,当参数的值不能与任何枚举成员匹配时,则构造失败:
1 2 3 4 5 6 7 8 9 10 11 |
|
带原始值的枚举类型的可失败构造器
带原始值的枚举类型会自带一个可失败构造器init?(rawValue:)
,该可失败构造器有一个名为rawvalue
的参数,其类型和枚举类型的原始值类型一致,如果该参数的值能够和某个枚举成员的原始值匹配,则该构造器会构造相应的枚举成员,否则构造失败.
因此上面的TemperatureUnit
的例子可以重写为:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
构造失败的传递
类,结构体,枚举的可失败构造器可以横向代理到类型中的其它可失败构造器。类似的,子类的可失败构造器也能向上代理到父类的可失败构造器。
无论是向上代理还是横向代理,如果你代理的其它可失败构造器触发构造失败,整个构造过程将立即停止,接下来的任何构造代码不会再被执行。
注意:
可失败构造器也可以代理到其它的非可失败构造器,通过这种方式,你可以增加一个可能的失败状态到现有的构造过程中
下面的例子中,定义了一个名为CartItem
的Product
类的子类,这个类建立了一个在线购物车中的物品的模型,它有一个名为quantity
的常量存储型属性,并确保该属性的值至少为1:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
|
CartItem
可失败构造器首先验证接收的quantity
值是否大于等于1.倘若quantity
值无效,则立即终止整个构造过程,返回失败结果,且不再执行余下代码。同样滴,Product
的可失败构造器首先检查name值,假如name
值为空字符串,则构造器立即执行失败。
如果你通过传入一个非空字符串name以及一个值大于等于1的quantity
来创建一个CartItem
实例,那么构造方法能够成功被执行:
1 2 3 4 |
|
倘若你以一个值为0的quantity
来创建一个CartItem
实例,那么将导致CartItem
构造器失败:
1 2 3 4 5 6 |
|
同样滴,如果你尝试传入一个值为空字符串的name来创建一个CartItem
实例,那么将导致父类Product
的构造过程失败:
1 2 3 4 5 6 |
|
重写一个可失败构造器
如同其他的构造器,你可以在子类中重写父类的可失败构造器,或者你也可以用子类的非可失败构造器重写一个父类的可失败构造器。这使你可以定义一个不会构造失败的子类,即使父类的构造器允许构造失败。
注意,当你用子类的非可失败构造器重写父类的可失败构造器时,向上代理到父类的可失败构造器的唯一方式是对父类的可失败构造器的返回值进行强制解包
注意
你可以用非可失败构造器重写可失败构造器,但反过来却不行
下例中定义了一个名为Document
的类,name属性的值必须为一个非空字符串或nil,但不能是一个空字符串:
1 2 3 4 5 6 7 8 9 10 |
|
下面的这个例子,定义了一个Document
类的子类AutomaticallyNamedDocument
。这个子类重写了父类的两个指定构造器,确保了无论使用init()构造器,还是使用init(name:)构造器并为参数传递空字符串。生成的实例中的name属性总有初始"Untitled";
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|
AutomaticallyNamedDocument
用一个非可失败构造器init(name:)
重写父类的课失败构造器(init?(name:))。因为子类用另一种方式处理了空字符串的情况,所以不再需要一个可失败构造器,因此子类用一个非可失败构造器代替了父类的可失败构造器。
你可以在子类的非可失败构造器中使用强制解包来调用父类的可失败构造器。比如:下面的UntitledDocument
子类的name属性的值总是"untitled",它在构造过程中使用了父类的可失败构造器init?(name:)
1 2 3 4 5 |
|
在这个例子中,如果在调用父类的可失败构造器init?(name:)时传入的是空字符串,那么强制解包操作会引发运行时错误。不过,因为这里是通过非空的字符串常量来调用它,所以并不会发生运行时错误
可失败构造器 init!
通常来说我们通过在init关键字后添加问号的方式(init?)来定义一个可失败构造器,但你也可以通过在init后面添加感叹号的方式来定义一个可失败构造器(init!),该可失败构造器将会构建一个对应类型的隐式解包可选类型的对象。
你可以在init?中代理到init!,反之亦然。你也可以用init?重写init!,反之亦然。你还可以用init代理到init!,不过一旦init!构造失效,则会触发一个断言。
必要构造器
在类的构造器前添加required
修饰符表明所有该类的子类都必须实现该构造器:
1 2 3 4 5 |
|
在子类重写父类的必要构造器时,必须在子类的构造前也添加required
修饰符,表明该构造器要求也应用于继承链后面的子类。在重写父类中必要的指定构造器时,不需要添加override修饰符:
1 2 3 4 5 |
|
注意
通过闭包或函数设置属性的默认值
如果某个存储型属性的默认值需要一些定制或设置,你可以使用必要或全局函数为其提供定制的默认值。每当某个属性所在类型的新实例被创建时,对应的必要或函数会被调用,而它们的返回值会当做默认值赋值给这个属性,
这种类型的闭包或函数通常会创建一个跟属性类型相同的临时变量,然后修改它的值以满足预期的初始状态,最后返回这个临时变量,作为属性的默认值。
下面介绍了如何用闭包为属性提供默认值:
1 2 3 4 5 6 7 |
|
注意闭包结尾的大括号后面接了一对空的小括号。这用来告诉Swift立即执行此比好。如果你忽略了这对括号,相当于将闭包本省作为值赋值给了属性,而不是将闭包的返回值赋值给属性。
注意:
如果你使用闭包来初始化属性,请记住在闭包执行时,实例的其它部分都还没有初始化,这意味着你不能在闭包里访问其它属性,即使这些属性都有默认值。同样,你也不能使用隐式的
self
属性,或者调用任何实例方法
下面例子中定义了一个结构体Checkerboard
,它构建了西洋跳棋游戏的棋盘:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
|
没当一个新的Checkerboard
实例被创建时,赋值闭包会被执行,boardColors
的默认值会被计算出来并返回。上面例子中描述的闭包将计算出棋盘中每个格子对应的颜色,并将这些值保存到一个临时数组temporaryBoard
中,最后在构建完成时将此数组作为闭包返回值返回。这个返回的数组会保存到boardColors
中,并可以通过工具函数
squareIsBlackAtRow
来查询:
1 2 3 4 5 |
|