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