開篇
相信大家在學習和使用Swift的時候,肯定會被 !
和 ?
搞瘋過万搔, 糾結(jié)這兩個符號到底是個什么鬼 胡桨?鬼知道什么時候使用!
官帘,什么時候使用?
下面就說一下!
和 ?
區(qū)別以及該怎么使用!
?
和 !
到底是個啥
?
和 !
其實分別是Swift語言中對一種可選類型( Optional) 操作的語法糖昧谊。
那可選類型是干什么的呢刽虹? Swift中是可以聲明一個沒有初始值的屬性, Swift中引入了可選類型(Optional)來解決這一問題揽浙。它的定義是通過在類型聲明后加一個 ?
操作符完成的状婶。
例如: var name: String?
Optional
其實是個enum
意敛,里面有None
和Some
兩種類型馅巷。其實所謂的nil就是Optional.None
, 非nil就是Optional.Some
草姻, 然后會通過Some(T)
包裝(wrap)原始值钓猬,這也是為什么在使用Optional
的時候要拆包(從enum
里取出來原始值)的原因。這里是enum Optional的定義
enum Optional<T> : LogicValue, Reflectable {
case None
case Some(T)
init()
init(_ some: T)
/// Allow use in a Boolean context.
func getLogicValue() -> Bool
/// Haskell's fmap, which was mis-named
func map<U>(f: (T) -> U) -> U?
func getMirror() -> Mirror
}
既然這樣撩独, 那對于 var name: String?
該怎樣去理解這句語法呢敞曹?
var name: String?
// 上面這個Optional的聲明,是”我聲明了一個Optional類型值综膀,
它可能包含一個String值澳迫,也可能什么都不包含”,
也就是說實際上我們聲明的是Optional類型剧劝,而不是聲明了一個String類型 (這其實理解起來挺蛋疼的...)
?
和 !
使用
一旦聲明為Optional
的橄登,如果不顯式的賦值就會有個默認值nil
。判斷一個Optional
的值是否有值讥此,可以用if來判斷:
if name {
// 有值再操作
}
怎么使用Optional
值呢拢锹?文檔中也有提到說,在使用Optional
值的時候需要在具體的操作萄喳,比如調(diào)用方法卒稳、屬性、下標索引等前面需要加上一個?
他巨,如果是nil
值充坑,也就是Optional.None
,會跳過后面的操作不執(zhí)行染突,如果有值捻爷,就是Optional.Some
,可能就會拆包(unwrap)觉痛,然后對拆包后的值執(zhí)行后面的操作役衡,來保證執(zhí)行這個操作的安全性。
// 例如: let length = name?.characters.count
PS:對于
Optional
值薪棒,不能直接進行操作手蝎,否則會報錯榕莺。
?
的使用場景:
1.聲明Optional
值變量
2.用在對Optional
值操作中,用來判斷是否能響應后面的操作
3.使用 as?
向下轉(zhuǎn)型(Downcast)
上面提到Optional
值需要拆包(unwrap)后才能得到原來值棵介,然后才能對其操作钉鸯,那怎么來拆包呢?
拆包有兩種方法:
-
可選綁定(Optional Binding)
可選綁定(Optional Binding)是一種更簡單更推薦的方法來解包一個可選類型邮辽。 使用可選綁定來檢查可選類型的變量有值還是沒值唠雕。如果有值, 解包它并且將值傳遞給一個常量或者變量。
// 例子最為簡單明了
var str: String? = "Hello"
let greeting = "World!"
if let name = str {
let message = greeting + name
print(message
)}
/**自然語言解釋意思:就是如果str有值,解包它,并且將它的值賦值給name,
然后執(zhí)行下面的條件語句; 如果str為空, 直接跳過條件語句塊吨述。*/
-
硬解包
硬解包即直接在可選類型后面加一個感嘆號(!)來表示它肯定有值岩睁。
var str1: String? = "Hello"
let greeting = "World!"
if (str1 != nil) {
let message = greeting + str1!
print(message)
}
/**上面例子,我們只是自己知道str1肯定有值, 所以才直接硬解包了str1變量揣云。 但是萬一有時候我們的感覺是錯的,
那程序在運行時可能會出現(xiàn)嚴重的錯誤. 所以Swift中是推薦先檢查可選類型是否有值, 然后再進行解包的!
*/
錯誤示范:var str1:String? // str1值可能是傳過來的值或者從服務器獲取的值
let msg = "Hi"
let txt = msg + str1! // runtime error
/** 以上代碼在編譯階段不會報錯.因為使用了硬解包, 編譯器認為可選類型是有值的, 所以編譯是通過的.
當代碼運行起來時, 知名的錯誤將會出現(xiàn): `fatal error: Can’t unwrap Optional.None`
/*
PS:對于
!
操作符捕儒,這里的變量值一定是非nil的!
其實邓夕, 還有一種叫隱式拆包(Implicitly Unwrapped Optionals)刘莹,比如 對于會在viewDidLoad進行初始化的變量,可以直接定義為var str :String焚刚!
等于說你每次對這種類型的值操作時点弯,都會自動在操作前補上一個!
進行拆包,然后在執(zhí)行后面的操作矿咕,當然如果該值是nil抢肛,會報錯crash掉。
舉個很淺顯的栗子:
// 在一個viewController里面痴腌,從xib里面拖一個UIImageView控件雌团, 你會發(fā)現(xiàn)Xcode會自動給你轉(zhuǎn)成下面的形式
@IBOutlet weak var headerBGImageView: UIImageView!
/** 聲明Implicitly Unwrapped Optionals值,一般用于類中的屬性*/
PS:如果你在隱式解析可選類型沒有值的時候進行取值士聪,會crash锦援。和在沒有值的可選類型里面拆包是一樣的。
!
的使用場景:
1.強制對Optional
值進行拆包(unwrap)
2.聲明隱式拆包變量剥悟,一般用于類中的屬性
結(jié)束
其實!
和 ?
的問題是很坑的灵寺,不要看它僅僅是兩個符號,因為只要有一個不小心区岗,不注意略板,你會發(fā)現(xiàn)項目運行起來,會莫名的crash掉了慈缔,關(guān)鍵是Debug模式也不是很方便定位錯誤類型叮称。 自己整理一下關(guān)于 可選類型的相關(guān)使用,一是記錄和鞏固所學,而是希望會對大家有所幫助瓤檐。 本文可能會有錯誤和不妥之處赂韵,還望提出,我會及時改正挠蛉。