首先我们先看下Objective-C与Swift语言对于可选nil的不同理解:
Objective-C中的nil:表示缺少一个合法的对象,是指向不存在对象的指针,对结构体、枚举等类型不起作用(会返回NSNotFound)
Swift中的nil:表示任意类型的值缺失,是一个确定的值,要么是该类型的一个值要么什么都没有(即为nil)
一、申明可选常量或变量
let status: Int? = // 申明可选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! = // 第一次申明有初始值
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 == { return nil }
self.someValue = someValue
}
} func returnPossibleClass(value: Bool) -> SomeClass? { // 返回的类实例可能为nil
if value {
return SomeClass(someValue: ) // 返回的为nil
} else {
return SomeClass(someValue: ) // 返回的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: ) // 实例化对象时不需要对possibleValue初始化,someClass实例中:someValue为4,possibleValue为nil
注意:类中所有属性都需要有默认值。属性可以在申明时赋予初始值,也可以在构造方法中赋予初始值;子类继承父类时,必须先给自身属性先初始化后再继承父类构造方法初始化。一般的,出于安全的因素,子类的属性都赋予初始值或直接定义为可选类型
。
B:可选类型在结构体中的运用
struct PossibleStruct {
var someValue: Int
var possibleValue: String? // 可选存储属性,默认值为nil
// 结构体中可以自定义一个init构造器对属性初始化,也可以不自定义
}
let someStruct = PossibleStruct(someValue: , possibleValue: nil)