首先我们先看下Objective-C与Swift语言对于可选nil的不同理解:
Objective-C中的nil:表示缺少一个合法的对象,是指向不存在对象的指针,对结构体、枚举等类型不起作用(会返回NSNotFound)
Swift中的nil:表示任意类型的值缺失,是一个确定的值,要么是该类型的一个值要么什么都没有(即为nil)
一、申明可选常量或变量
let status: Int? = 1 // 申明可选Int类型的常量,初始值为1 var defaultAddress: String? = "上海徐汇" // 申明可选String类型的变量,初始值为"上海徐汇" var str: String? // 申明可选String(自定义的类)的变量,初始值为nil
注意:Int?
与Int
不相同,Int?
表示可选的Int类型,可以赋值为nil,而Int
不可以赋值为nil
二、使用"!"强制解析获取可选类型的值(不建议直接使用)
var defaultAddress: String? = "上海徐汇" if defaultAddress != nil { // !=或==可以用来判断是否为nil print("您的地址是\(defaultAddress!)") // 使用!强制解析 } else { print("对不起,您不存在地址信息") } //您的地址是上海徐汇 var str: String? print("str为\(str!)") //Unexpectedly found nil while unwrapping an Optional value(XCode会提示运行错误,因为str初始值为nil,强制解析不行)
三、使用可选绑定获取可选类型的值(建议的用法)
var defaultAddress: String? = "上海徐汇" if let address = defaultAddress { // 如果defaultAddress有值或类型转换成功,则将值赋值给address直接使用 print("您的地址是\(address)") // 使用address代替defaultAddress,且不需要加!强制解析 } else { print("对不起,您不存在地址信息") } //您的地址是上海徐汇
四、隐式解析可选类型(用于申明时肯定有初始值,但后面可能为nil)
var mobileNumber: Int64! = 13812345678 // 第一次申明有初始值 print("您的电话号码是\(mobileNumber)") // 不需要使用!强制解析 // 您的电话号码是Optional(13812345678) // 还是不建议直接强制解析,因为实际项目中可能中间已经对该值做了改变,若为nil则会运行错误导致APP崩溃 if let number = mobileNumber { // 建议的做法 print("您的电话号码是\(number)") // 打印内容:**您的电话号码是****13812345678** } else { print("您没有记录电话号码") } //您的电话号码是13812345678
五、空合运算符(用于判断变量或常量是否为nil)
空合运算符(a ?? b
)将对可选类型a
进行空判断,如果a
包含一个值就进行解封,否则就返回一个默认值b
.这个运算符有两个条件:
- 表达式
a
必须是Optional类型 - 默认值
b
的类型必须要和a
存储值的类型保持一致
空合运算符是对以下代码的简短表达方法
a != nil ? a! : b
上述代码使用了三目运算符。当可选类型a
的值不为空时,进行强制解封(a!
)访问a
中值,反之当a
中值为空时,返回默认值b。无疑空合运算符(??
)提供了一种更为优雅的方式去封装条件判断和解封两种行为,显得简洁以及更具可读性
下文例子采用空合运算符,实现了在默认颜色名和可选自定义颜色名之间抉择:
let defaultColorName = "red" var userDefinedColorName: String? //默认值为 nil var colorNameToUse = userDefinedColorName ?? defaultColorName // userDefinedColorName 的值为空,所以 colorNameToUse 的值为 "red"
六、函数/方法返回类型为可选类型
A:返回值为可选类型的值(如Int?、String?、(Int, String)?、[Int]?、[Int: String]?等)
func returnPossibleValue(value: Bool) -> String? { // 返回类型为可选String类型 if value { return "返回类型是可选类型值" // 如果为真,返回Int类型的值1 } else { return nil // 如果为假,返回nil } } let possibleValue = returnPossibleValue(value: true) // 要用可选绑定判断再使用,因为possibleValue为String?可选类型 if let value = possibleValue { print(value) } else { print("none value") }
B:返回值为可选类型的类(如URL?、自定义Person?等)
class SomeClass { var someValue: Int init?(someValue: Int) { // 可失败构造器 if someValue == 1 { return nil } self.someValue = someValue } } func returnPossibleClass(value: Bool) -> SomeClass? { // 返回的类实例可能为nil if value { return SomeClass(someValue: 1) // 返回的为nil } else { return SomeClass(someValue: 2) // 返回的SomeClass?实例,不为nil } }
C:返回值为可选类型的闭包(如(()-> (void))? )
func returnOptionalFunc(value: Bool) -> (() -> (Void))? { // 返回类型为可选类型的闭包 if value { return { () in print("返回类型是可选类型闭包") } } else { return nil } } let possibleFunc = returnOptionalFunc(value: true) // 要用可选绑定判断再使用,因为possibleFunc 为可选类型的闭包,类型为() -> (Void) if let aFunc = possibleFunc { print(aFunc()) // 注意增加()调用闭包,因为没有参数则是空括号 } else { print("none func") }
七、可选类型在类或结构体中的运用
A:可选类型在类中的运用
class PossibleClass { var someValue: Int var possibleValue: String? // 可选存储属性,默认值为nil init(someValue: Int) { // 构造方法中可以不对possibleValue属性初始化 self.someValue = someValue } } let someClass = PossibleClass(someValue: 4) // 实例化对象时不需要对possibleValue初始化,someClass实例中:someValue为4,possibleValue为nil
注意:类中所有属性都需要有默认值。属性可以在申明时赋予初始值,也可以在构造方法中赋予初始值;子类继承父类时,必须先给自身属性先初始化后再继承父类构造方法初始化。一般的,出于安全的因素,子类的属性都赋予初始值或直接定义为可选类型
。
B:可选类型在结构体中的运用
struct PossibleStruct { var someValue: Int var possibleValue: String? // 可选存储属性,默认值为nil // 结构体中可以自定义一个init构造器对属性初始化,也可以不自定义 } let someStruct = PossibleStruct(someValue: 4, possibleValue: nil)