Swift - 類型轉(zhuǎn)換

類型轉(zhuǎn)換 可以判斷實例的類型,也可以將實例看做是其父類或者子類的實例陵珍。

類型轉(zhuǎn)換在 Swift 中使用 isas 操作符實現(xiàn)违施。這兩個操作符提供了一種簡單達意的方式去檢查值的類型或者轉(zhuǎn)換它的類型醉拓。

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

定義一個類層次作為例子

你可以將類型轉(zhuǎn)換用在類和子類的層次結(jié)構(gòu)上秆乳,檢查特定類實例的類型并且轉(zhuǎn)換這個類實例的類型成為這個層次結(jié)構(gòu)中的其他類型钻哩。下面的三個代碼段定義了一個類層次和一個包含了這些類實例的數(shù)組街氢,作為類型轉(zhuǎn)換的例子。

第一個代碼片段定義了一個新的基類 MediaItem荣刑。這個類為任何出現(xiàn)在數(shù)字媒體庫的媒體項提供基礎功能伦乔。特別的烈和,它聲明了一個 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)
    }
}

最后一個代碼段創(chuàng)建了一個數(shù)組常量 library,包含兩個 Movie 實例和三個 Song 實例砰逻。library 的類型是在它被初始化時根據(jù)它數(shù)組中所包含的內(nèi)容推斷來的鸣驱。Swift 的類型檢測器能夠推斷出 MovieSong 有共同的父類 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")
]
// 數(shù)組 library 的類型被推斷為 [MediaItem]

在幕后 library 里存儲的媒體項依然是 MovieSong 類型的蝠咆。但是踊东,若你迭代它,依次取出的實例會是 MediaItem 類型的刚操,而不是 MovieSong 類型闸翅。為了讓它們作為原本的類型工作,你需要檢查它們的類型或者向下轉(zhuǎn)換它們到其它類型坚冀,就像下面描述的一樣。

檢查類型

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

下面的例子定義了兩個變量,movieCountsongCount豁遭,用來計算數(shù)組 libraryMovieSong 類型的實例數(shù)量:

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”

向下轉(zhuǎn)型

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

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

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

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

下面的例子捅厂,迭代了 library 里的每一個 MediaItem贯卦,并打印出適當?shù)拿枋觥R@樣做焙贷,item 需要真正作為 MovieSong 的類型來使用脸侥,而不僅僅是作為 MediaItem。為了能夠在描述中使用 MovieSongdirectorartist 屬性盈厘,這是必要的睁枕。

在這個示例中,數(shù)組中的每一個 item 可能是 MovieSong沸手。事前你不知道每個 item 的真實類型外遇,所以這里使用條件形式的類型轉(zhuǎn)換(as?)去檢查循環(huán)里的每次下轉(zhuǎn):

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

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

AnyAnyObject 的類型轉(zhuǎn)換

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

  • Any 可以表示任何類型,包括函數(shù)類型捐晶。
  • AnyObject 可以表示任何類類型的實例菲语。

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

這里有個示例山上,使用 Any 類型來和混合的不同類型一起工作,包括函數(shù)類型和非類類型英支。它創(chuàng)建了一個可以存儲 Any 類型的數(shù)組 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 數(shù)組包含兩個 Int 值佩憾,兩個 Double 值,一個 String 值,一個元組 (Double, Double)妄帘,一個Movie實例“Ghostbusters”楞黄,以及一個接受 String 值并返回另一個 String 值的閉包表達式。

你可以在 switch 表達式的 case 中使用 isas 操作符來找出只知道是 AnyAnyObject 類型的常量或變量的具體類型抡驼。下面的示例迭代 things 數(shù)組中的每一項鬼廓,并用 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操作符顯式轉(zhuǎn)換為Any眷篇,如下所示:

let optionalNumber: Int? = 3
things.append(optionalNumber)        // 警告
things.append(optionalNumber as Any) // 沒有警告
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末萎河,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子蕉饼,更是在濱河造成了極大的恐慌虐杯,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,039評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件昧港,死亡現(xiàn)場離奇詭異擎椰,居然都是意外死亡,警方通過查閱死者的電腦和手機创肥,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評論 3 395
  • 文/潘曉璐 我一進店門达舒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人叹侄,你說我怎么就攤上這事巩搏。” “怎么了趾代?”我有些...
    開封第一講書人閱讀 165,417評論 0 356
  • 文/不壞的土叔 我叫張陵贯底,是天一觀的道長。 經(jīng)常有香客問我撒强,道長禽捆,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,868評論 1 295
  • 正文 為了忘掉前任飘哨,我火速辦了婚禮胚想,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘杖玲。我一直安慰自己顿仇,他們只是感情好,可當我...
    茶點故事閱讀 67,892評論 6 392
  • 文/花漫 我一把揭開白布摆马。 她就那樣靜靜地躺著臼闻,像睡著了一般。 火紅的嫁衣襯著肌膚如雪囤采。 梳的紋絲不亂的頭發(fā)上述呐,一...
    開封第一講書人閱讀 51,692評論 1 305
  • 那天,我揣著相機與錄音蕉毯,去河邊找鬼乓搬。 笑死,一個胖子當著我的面吹牛代虾,可吹牛的內(nèi)容都是我干的进肯。 我是一名探鬼主播,決...
    沈念sama閱讀 40,416評論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼棉磨,長吁一口氣:“原來是場噩夢啊……” “哼江掩!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起乘瓤,我...
    開封第一講書人閱讀 39,326評論 0 276
  • 序言:老撾萬榮一對情侶失蹤环形,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后衙傀,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體抬吟,經(jīng)...
    沈念sama閱讀 45,782評論 1 316
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,957評論 3 337
  • 正文 我和宋清朗相戀三年统抬,在試婚紗的時候發(fā)現(xiàn)自己被綠了火本。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,102評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡聪建,死狀恐怖发侵,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情妆偏,我是刑警寧澤刃鳄,帶...
    沈念sama閱讀 35,790評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站钱骂,受9級特大地震影響叔锐,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜见秽,卻給世界環(huán)境...
    茶點故事閱讀 41,442評論 3 331
  • 文/蒙蒙 一愉烙、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧解取,春花似錦步责、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,996評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽遂鹊。三九已至,卻和暖如春蔗包,著一層夾襖步出監(jiān)牢的瞬間秉扑,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,113評論 1 272
  • 我被黑心中介騙來泰國打工调限, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留舟陆,地道東北人。 一個月前我還...
    沈念sama閱讀 48,332評論 3 373
  • 正文 我出身青樓耻矮,卻偏偏與公主長得像秦躯,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子裆装,可洞房花燭夜當晚...
    茶點故事閱讀 45,044評論 2 355

推薦閱讀更多精彩內(nèi)容