从 Objective-C 转 Swift 开发已经有一段时间了,这两门语言在整体的理念上差异还是蛮大的。在这之中,可选类型的处理是每一个使用 Swift 的开发者每天都要面临的问题,理解并正确处理好可选类型对于写出高质量的 Swift 代码和保证 iOS 项目的健壮性都是至关重要的。
可选类型
要想处理好可选类型,就要先理解可选类型。
一个可选类型代表有两种可能性:有一个值,你可以解包可选类型来访问该值;或者根本没有值。
在 Objective-C 中不存在可选类型的概念,Objective-C 中最接近的东西就是 nil, nil 的意思是“没有有效的对象”。但是,这只适用于对象——它不适用于结构、基本数据类型或枚举值。对于这些类型,Objective-C 方法通?;岱祷匾桓鎏厥庵担ㄈ?NSNotFound)来指示缺少值。这种方法假设方法的调用者知道有一个特殊的值来测试,并记得检查它。Swift 的可选值可以让你指出可能为 nil 的任何类型的值,而不需要特殊的常量。
例如,Swift 的 Int 类型有一个初始化方法,它试图将一个 String 值转换成一个 Int 值。但是,并不是每个字符串都可以转换成一个整数。字符串 "123" 可以转换为数字值 123,但字符串 "Hello, world" 没有一个明显的数值要转换。
下面的例子使用初始化方法来尝试将一个字符串转换为一个 Int:
let possibleNumber = "123"
let convertedNumber = Int(possibleNumber)
// convertedNumber 被推断为 "Int?" 类型或 “可选的 Int”
nil
通过赋值给它一个特殊的值 nil 来设置一个可选变量为无值状态:
var serverResponseCode: Int? = 404
// serverResponseCode 包含一个实际的 Int 值为 404
serverResponseCode = nil
// serverResponseCode 现在不包含任何值
如果你定义了一个可选变量而不提供默认值,则该变量会自动设置为 nil:
var surveyAnswer: String?
// surveyAnswer 自动设置为 nil
Swift 的 nil 与 Objective-C 中的 nil 不相同。在 Objective-C 中,nil 是一个指向不存在对象的指针。在 Swift 中,nil 不是一个指针,它是缺少某种类型的值。任何类型的可选值都可以被设置为 nil,而不仅仅是对象类型。
可选绑定
你可以使用可选绑定来发现可选值是否包含值,如果有,则使用该值用作临时常量或变量。可选绑定可以与 if 和 while 语句一起使用,以检查可选值内部的值,并将该值提取为常量或变量,作为单次操作的一部分。
使用 if 语句编写一个可选绑定,如下所示:
if let actualNumber = Int(possibleNumber) {
print("\"\(possibleNumber)\" has an integer value of \(actualNumber)")
} else {
print("\"\(possibleNumber)\" could not be converted to an integer")
}
// 打印 ""123" has an integer value of 123"
如果转换成功,那么 actualNumber 常量可以在 if 语句的第一个分支中使用。它已经被初始化为包含在非可选的值中,所以没有必要使用 ! 后缀来访问它的值。
你可以使用可选绑定的常量和变量。如果你想在 if 语句的第一个分支内操作 actualNumber 的值,你可以写 if var actualNumber,使得可选值作为一个变量而非常量。
你可以根据需要在单个 if 语句中包含尽可能多的可选绑定和布尔条件,并用逗号分隔。如果可选绑定中的任何值为 nil,或者任何布尔条件的计算结果为 false,则整个 if 语句的条件被认为是错误的。以下 if 语句是等价的:
if let firstNumber = Int("4"), let secondNumber = Int("42"), firstNumber < secondNumber && secondNumber < 100 {
print("\(firstNumber) < \(secondNumber) < 100")
}
// 打印 "4 < 42 < 100"
if let firstNumber = Int("4") {
if let secondNumber = Int("42") {
if firstNumber < secondNumber && secondNumber < 100 {
print("\(firstNumber) < \(secondNumber) < 100")
}
}
}
// 打印 "4 < 42 < 100"
隐式解包可选类型
有时从程序的结构中可以清楚的看到,在第一次设置值之后,可选值将始终有一个值。在这些情况下,每次访问时都不需要检查和解包可选值,因为可以安全地假定所有的时间都有一个值。
这些可选值被定义为隐式解包可选值。你写一个隐式解包的可选值,在你想要的可选类型之后放置一个感叹号(String!)而不是一个问号(String?)
隐式解包可选值的背后是普通可选值,但也可以像非可选值一样使用,而不必在每次访问时解包可选值。
let possibleString: String? = "An optional string."
let forcedString: String = possibleString! // 需要感叹号
let assumedString: String! = "An implicitly unwrapped optional string."
let implicitString: String = assumedString // 不需要感叹号
如果隐式解包可选值为 nil,并且你尝试访问其包装的值,则会触发运行时错误。
你仍然可以对隐式解包可选值使用强制解包和可选绑定。