Swift中 !和 ?的區別及使用

wyixin 7年前發布 | 6K 次閱讀 Swift Apple Swift開發

開篇

相信大家在學習和使用Swift的時候,肯定會被 ! 和 ? 搞瘋過, 糾結這兩個符號到底是個什么鬼 ?鬼知道什么時候使用 ! ,什么時候使用 ?

WTF

下面就說一下 ! 和 ? 區別以及該怎么使用!

? 和 ! 到底是個啥

? 和 ! 其實分別是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 值的時候需要在具體的操作,比如調用方法、屬性、下標索引等前面需要加上一個 ? ,如果是 nil 值,也就是 Optional.None ,會跳過后面的操作不執行,如果有值,就是 Optional.Some ,可能就會拆包(unwrap),然后對拆包后的值執行后面的操作,來保證執行這個操作的安全性。

// 例如: let length = name?.characters.count

PS:對于 Optional 值,不能直接進行操作,否則會報錯。

? 的使用場景:

1.聲明 Optional 值變量

2.用在對 Optional 值操作中,用來判斷是否能響應后面的操作

3.使用 as? 向下轉型(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, 然后執行下面的條件語句; 如果str為空, 直接跳過條件語句塊。/</code></pre>

  • 硬解包

    硬解包即直接在可選類型后面加一個感嘆號(!)來表示它肯定有值。

    再舉栗子

var str1: String? = "Hello"
let greeting = "World!"
if (str1 != nil) { 
       let message = greeting + str1! 
       print(message)
}

/*上面例子,我們只是自己知道str1肯定有值, 所以才直接硬解包了str1變量。 但是萬一有時候我們的感覺是錯的, 那程序在運行時可能會出現嚴重的錯誤. 所以Swift中是推薦先檢查可選類型是否有值, 然后再進行解包的!/</code></pre>

錯誤示范:

這是:negative_squared_cross_mark: 滴

var str1:String?  // str1值可能是傳過來的值或者從服務器獲取的值
let msg = "Hi"
let txt = msg + str1!  // runtime error

/* 以上代碼在編譯階段不會報錯.因為使用了硬解包, 編譯器認為可選類型是有值的, 所以編譯是通過的. 當代碼運行起來時, 知名的錯誤將會出現: fatal error: Can’t unwrap Optional.None /</code></pre>

PS:對于 ! 操作符,這里的變量值一定是非nil的!

其實, 還有一種叫隱式拆包(Implicitly Unwrapped Optionals),比如 對于會在viewDidLoad進行初始化的變量,可以直接定義為 var str :String! 等于說你每次對這種類型的值操作時,都會自動在操作前補上一個 ! 進行拆包,然后在執行后面的操作,當然如果該值是nil,會報錯crash掉。

舉個很淺顯的栗子:

最后一個栗子

// 在一個viewController里面,從xib里面拖一個UIImageView控件, 你會發現Xcode會自動給你轉成下面的形式 
@IBOutlet weak var headerBGImageView: UIImageView!

/* 聲明Implicitly Unwrapped Optionals值,一般用于類中的屬性/</code></pre>

PS:如果你在隱式解析可選類型沒有值的時候進行取值,會crash。和在沒有值的可選類型里面拆包是一樣的。

! 的使用場景:

1.強制對 Optional 值進行拆包(unwrap)

2.聲明隱式拆包變量,一般用于類中的屬性

結束

其實 ! 和 ? 的問題是很坑的,不要看它僅僅是兩個符號,因為只要有一個不小心,不注意,你會發現項目運行起來,會莫名的crash掉了,關鍵是Debug模式也不是很方便定位錯誤類型。 自己整理一下關于 可選類型的相關使用,一是記錄和鞏固所學,而是希望會對大家有所幫助。 本文可能會有錯誤和不妥之處,還望提出,我會及時改正。

 

來自:http://www.jianshu.com/p/89a2afb82488

 

 本文由用戶 wyixin 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!