iOS 學習 Swift 教程- 2.19 類型轉換

前言:
本篇文章的目的,在于記錄學習過程,敦促自己,方便查看。

練習工具:Playground
學習網站: swift51

本頁包含內容:

類型轉換 可以判斷實例的類型,也可以將實例看做是其父類或者子類的實例。

類型轉換在 Swift 中使用 is 和 as 操作符實現。這兩個操作符提供了一種簡單達意的方式去檢查值的類型或者轉換它的類型。

你也可以用它來檢查一個類型是否實現了某個協議,就像在檢驗協議的一致性部分講述的一樣。

定義一個類層次作為例子

你可以將類型轉換用在類和子類的層次結構上,檢查特定類實例的類型并且轉換這個類實例的類型成為這個層次結構中的其他類型。下面的三個代碼段定義了一個類層次和一個包含了這些類實例的數組,作為類型轉換的例子。

第一個代碼片段定義了一個新的基類 MediaItem。這個類為任何出現在數字媒體庫的媒體項提供基礎功能。特別的,它聲明了一個 String 類型的 name 屬性,和一個 init(name:) 初始化器。(假定所有的媒體項都有個名稱。)

 class MediaItem {
     var name: String
     init(name: String) {
         self.name = name
     }
 }

下一個代碼段定義了 MediaItem 的兩個子類。第一個子類 Movie 封裝了與電影相關的額外信息,在父類(或者說基類)的基礎上增加了一個 director(導演)屬性,和相應的初始化器。第二個子類 Song,在父類的基礎上增加了一個 artist(藝術家)屬性,和相應的初始化器:

 class Movie: MediaItem {
     var director: String
     init(name: String, director: String) {
         self.director = director
         super.init(name: name)
     }
 }
 
 class Song: MediaItem {
     var artist: String
     init(name: String, artist: String) {
         self.artist = artist
         super.init(name: name)
     }
 }

最后一個代碼段創建了一個數組常量 library,包含兩個 Movie 實例和三個 Song 實例。library 的類型是在它被初始化時根據它數組中所包含的內容推斷來的。Swift 的類型檢測器能夠推斷出 Movie 和 Song 有共同的父類 MediaItem,所以它推斷出 [MediaItem] 類作為 library 的類型:

 let library = [
    Movie(name: "Casablanca", director: "Michael Curtiz"),
    Song(name: "Blue Suede Shoes", artist: "Elvis Presley"),
    Movie(name: "Citizen Kane", director: "Orson Welles"),
    Song(name: "The One And Only", artist: "Chesney Hawkes"),
    Song(name: "Never Gonna Give You Up", artist: "Rick Astley")
]
// 數組 library 的類型被推斷為 [MediaItem]

在幕后 library 里存儲的媒體項依然是 Movie 和 Song 類型的。但是,若你迭代它,依次取出的實例會是 MediaItem 類型的,而不是 Movie 和 Song 類型。為了讓它們作為原本的類型工作,你需要檢查它們的類型或者向下轉換它們到其它類型,就像下面描述的一樣。

檢查類型

用類型檢查操作符(is)來檢查一個實例是否屬于特定子類型。若實例屬于那個子類型,類型檢查操作符返回 true,否則返回 false。

下面的例子定義了兩個變量,movieCount 和 songCount,用來計算數組 library 中 Movie 和 Song 類型的實例數量:

 var movieCount = 0
 var songCount = 0
 
 for item in library {
     if item is Movie {
         movieCount += 1
     } else if item is Song {
         songCount += 1
     }
 }

print("Media library contains \(movieCount) movies and \(songCount) songs")
// 打印 “Media library contains 2 movies and 3 songs”

示例迭代了數組 library 中的所有項。每一次,for-in 循環設置 item 為數組中的下一個 MediaItem。

若當前 MediaItem 是一個 Movie 類型的實例,item is Movie 返回 true,否則返回 false。同樣的,item is Song 檢查 item 是否為 Song 類型的實例。在循環結束后,movieCount 和 songCount 的值就是被找到的屬于各自類型的實例的數量。

向下轉型

某類型的一個常量或變量可能在幕后實際上屬于一個子類。當確定是這種情況時,你可以嘗試向下轉到它的子類型,用類型轉換操作符(as? 或 as!)。

因為向下轉型可能會失敗,類型轉型操作符帶有兩種不同形式。條件形式as? 返回一個你試圖向下轉成的類型的可選值。強制形式 as! 把試圖向下轉型和強制解包轉換結果結合為一個操作。

當你不確定向下轉型可以成功時,用類型轉換的條件形式(as?)。條件形式的類型轉換總是返回一個可選值,并且若下轉是不可能的,可選值將是 nil。這使你能夠檢查向下轉型是否成功。

只有你可以確定向下轉型一定會成功時,才使用強制形式(as!)。當你試圖向下轉型為一個不正確的類型時,強制形式的類型轉換會觸發一個運行時錯誤。

下面的例子,迭代了 library 里的每一個 MediaItem,并打印出適當的描述。要這樣做,item 需要真正作為 Movie 或 Song 的類型來使用,而不僅僅是作為 MediaItem。為了能夠在描述中使用 Movie 或 Song 的 director 或 artist 屬性,這是必要的。

在這個示例中,數組中的每一個 item 可能是 Movie 或 Song。事前你不知道每個 item 的真實類型,所以這里使用條件形式的類型轉換(as?)去檢查循環里的每次下轉:

 for item in library {
     if let movie = item as? Movie {
         print("Movie: '\(movie.name)', dir. \(movie.director)")
     } else if let song = item as? Song {
         print("Song: '\(song.name)', by \(song.artist)")
     }
 }
 
 // Movie: 'Casablanca', dir. Michael Curtiz
 // Song: 'Blue Suede Shoes', by Elvis Presley
 // Movie: 'Citizen Kane', dir. Orson Welles
 // Song: 'The One And Only', by Chesney Hawkes
 // Song: 'Never Gonna Give You Up', by Rick Astley

示例首先試圖將 item 下轉為 Movie。因為 item 是一個 MediaItem 類型的實例,它可能是一個 Movie;同樣,它也可能是一個 Song,或者僅僅是基類 MediaItem。因為不確定,as? 形式在試圖下轉時將返回一個可選值。item as? Movie 的返回值是 Movie? 或者說“可選 Movie”。

當向下轉型為 Movie 應用在兩個 Song 實例時將會失敗。為了處理這種情況,上面的例子使用了可選綁定(optional binding)來檢查可選 Movie 真的包含一個值(這個是為了判斷下轉是否成功。)可選綁定是這樣寫的“if let movie = item as? Movie”,可以這樣解讀:

“嘗試將 item 轉為 Movie 類型。若成功,設置一個新的臨時常量 movie 來存儲返回的可選 Movie 中的值”

若向下轉型成功,然后 movie 的屬性將用于打印一個 Movie 實例的描述,包括它的導演的名字 director。相似的原理被用來檢測 Song 實例,當 Song 被找到時則打印它的描述(包含 artist 的名字)。

注意
轉換沒有真的改變實例或它的值。根本的實例保持不變;只是簡單地把它作為它被轉換成的類型來使用。

Any 和 AnyObject 的類型轉換

Swift 為不確定類型提供了兩種特殊的類型別名:

 - Any 可以表示任何類型,包括函數類型。
 - AnyObject 可以表示任何類類型的實例。

只有當你確實需要它們的行為和功能時才使用 Any 和 AnyObject。在你的代碼里使用你期望的明確類型總是更好的。

這里有個示例,使用 Any 類型來和混合的不同類型一起工作,包括函數類型和非類類型。它創建了一個可以存儲 Any 類型的數組 things:

 var things = [Any]()
 
 things.append(0)
 things.append(0.0)
 things.append(42)
 things.append(3.14159)
 things.append("hello")
 things.append((3.0, 5.0))
 things.append(Movie(name: "Ghostbusters", director: "Ivan Reitman"))
 things.append({ (name: String) -> String in "Hello, \(name)" })

things 數組包含兩個 Int 值,兩個 Double 值,一個 String 值,一個元組 (Double, Double),一個Movie實例“Ghostbusters”,以及一個接受 String 值并返回另一個 String 值的閉包表達式。

你可以在 switch 表達式的 case 中使用 is 和 as 操作符來找出只知道是 Any 或 AnyObject 類型的常量或變量的具體類型。下面的示例迭代 things 數組中的每一項,并用 switch 語句查找每一項的類型。有幾個 switch 語句的 case 綁定它們匹配到的值到一個指定類型的常量,從而可以打印這些值:

數組中的每一項,并用 switch 語句查找每一項的類型。有幾個 switch 語句的 case 綁定它們匹配到的值到一個指定類型的常量,從而可以打印這些值:

 for thing in things {
     switch thing {
     case 0 as Int:
         print("zero as an Int")
     case 0 as Double:
         print("zero as a Double")
     case let someInt as Int:
         print("an integer value of \(someInt)")
     case let someDouble as Double where someDouble > 0:
         print("a positive double value of \(someDouble)")
     case is Double:
         print("some other double value that I don't want to print")
     case let someString as String:
         print("a string value of \"\(someString)\"")
     case let (x, y) as (Double, Double):
         print("an (x, y) point at \(x), \(y)")
     case let movie as Movie:
         print("a movie called '\(movie.name)', dir. \(movie.director)")
     case let stringConverter as (String) -> String:
   print(stringConverter("Michael"))
     default:
         print("something else")
     }
 }
 
 // zero as an Int
 // zero as a Double
 // an integer value of 42
 // a positive double value of 3.14159
 // a string value of "hello"
 // an (x, y) point at 3.0, 5.0
 // a movie called 'Ghostbusters', dir. Ivan Reitman
 // Hello, Michael

注意
Any類型可以表示所有類型的值,包括可選類型。Swift 會在你用Any類型來表示一個可選值的時候,給你一個警告。如果你確實想使用Any類型來承載可選值,你可以使用as操作符顯式轉換為Any,如下所示:

let optionalNumber: Int? = 3
things.append(optionalNumber)        // 警告
things.append(optionalNumber as Any) // 沒有警告
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,565評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,115評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,577評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,514評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,234評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,621評論 1 326
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,641評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,822評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,380評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,128評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,319評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,879評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,548評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,970評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,229評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,048評論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,285評論 2 376

推薦閱讀更多精彩內容