Swift類型檢查與轉換
繼承會發生在子類和父類中,如圖所示,是一係列類的繼承關係類圖,Person是類層次結構中的根類,Student是Person的直接子類,Worker是Person的直接子類。這個繼承關係類圖的具體實現代碼如下:
class Person { var name : String var age : Int func description() -> String { return "\(name) 年齡是: \(age)" } convenience init () { self.init(name: "Tony") self.age = 18 } convenience init (name : String) { self.init(name: name, age: 18) } init (name : String, age : Int) { self.name = name self.age = age } } class Student : Person { var school : String init (name : String, age : Int, school : String) { self.school = school super.init(name : name, age : age) } } class Worker : Person { var factory : String init (name : String, age : Int, factory : String) { self.factory = factory super.init(name : name, age : age) } }

下麵我們將以此為例,介紹Swift類的類型檢查與轉換,其中包括is操作符、as操作符以及Any和AnyObject類型等。
一、使用is操作符
is操作符可以判斷一個實例是否是某個類的類型。如果實例是目標類型,結果返回true,否則為false。
下麵看一個示例:
let Student1 = Student(name : "Tom", age : 18, school : "清華大學") ① let Student2 = Student(name : "Ben", age : 28, school : "北京大學") let Student3 = Student(name : "Tony", age : 38, school : "香港大學") ② let Worker1 = Worker(name : "Tom", age : 18, factory : "鋼廠") ③ let Worker2 = Worker(name : "Ben", age : 20, factory : "電廠") ④ let people = [Student1, Student2, Student3, Worker1, Worker2] ⑤ var StudentCount = 0 var WorkerCount = 0 for item in people { ⑥ if item is Worker { ⑦ ++WorkerCount } else if item is Student { ⑧ ++StudentCount } } println("工人人數:\(WorkerCount) ,學生人數:\(StudentCount) 。")
上述代碼第①行和第②行創建了3個Student實例,第③行和第④行創建了兩個Worker實例,然後把這5個實例放入people數組集合中。
在第⑥行使用for in遍曆people數組集合。在循環體中,第⑦行item is Worker表達式是判斷集合中的元素是否是Worker類的實例。類似地,第⑧行item is Student表達式是判斷集合中的元素是否是Student類的實例。
輸出結果如下:
工人人數:2,學生人數:3。
二、使用as操作符
在介紹as操作符之前,我們先了解一下對象的類型轉換,並不是所有的類型都能互相轉換。下麵先看如下語句:
let p1 : Person = Student(name : "Tom", age : 20, school : "清華大學")
let p2 : Person = Worker(name : "Tom", age : 18, factory : "鋼廠")
let p3 : Person = Person(name : "Tom", age : 28)
我們創建了3個實例p1、p2、p3,類型都是Person。p1是Student實例,p2是Worker實例,p3是Person實例。首先,對象類型轉換一定發生在繼承的前提下,p1和p2都聲明為Person類型,而實例是由Person子類型實例化的。

作為這段程序的編寫者,我們知道p1本質上是Student實例,但是表麵上看是Person類型,編譯器也無法推斷p1的實例是Person、Student還是Worker。我們可以使用is操作符來判斷它是哪一類的實例。然後在轉換時可以使用as操作符將其轉換為子類類型,即把Person類型的p1轉為Student子類類型,這種轉換被稱為向下轉型。這種轉換是有風險的,如果p1不是目標類型,轉換就會失敗。為了不發生異常,我們可以使用as?將其轉換為目標類型的可選類型,能夠成功則轉換,不成功則返回nil。
從表15-1可見,p1到Student類型轉換是向下轉型,能夠成功,p1到Worker類型轉換是向下轉型但會失敗,p1到Person類型轉換不需要向下轉型就能夠成功賦值,因為p1本身就是Person類型。p2與p1類似。
p3與p1和p2有很大的不同,因為p3本質上是Person實例,不能向下轉型。
下麵看一個示例:
let Student1 = Student(name : "Tom", age : 18, school : "清華大學") ① let Student2 = Student(name : "Ben", age : 28, school : "北京大學") let Student3 = Student(name : "Tony", age : 38, school : "香港大學") ② let Worker1 = Worker(name : "Tom", age : 18, factory : "鋼廠") ③ let Worker2 = Worker(name : "Ben", age : 20, factory : "電廠") ④ let people = [Student1, Student2, Student3, Worker1, Worker2] ⑤ for item in people { ⑥ if let Student = item as? Student { ⑦ println("Student school: \(Student.school)") ⑧ } else if let Worker = item as? Worker { ⑨ println("Worker factory: \(Worker.factory)") ⑩ } }
上述代碼第①行和第②行創建了3個Student實例,第③行和第④行創建了兩個Worker實例。然後把這5個實例放入people數組集合中。
在第⑥行使用for in遍曆people數組集合。在循環體中,第⑦行let Student = item as? Student語句使用as?操作符將元素轉換為Student類型。如果轉換成功,則把元素賦值給Student變量,否則將nil賦值給Student變量,轉換成功執行第⑧行代碼。第⑨行代碼與第⑦行代碼類似,不再贅述。
最後輸出結果如下:
Student school: 清華大學
Student school: 北京大學
Student school: 香港大學
Worker factory: 鋼廠
Worker factory: 電廠
三、使用Any和AnyObject類型
在Swift中還提供了兩種類型表示不確定類型:AnyObject和Any。AnyObject可以表示任何類的實例,而Any可以表示任何類型,包括類和其他數據類型,也包括Int和Double的基本數據類型。
下麵將上一節的示例修改如下:
let Student1 = Student(name : "Tom", age : 18, school : "清華大學") let Student2 = Student(name : "Ben", age : 28, school : "北京大學") let Student3 = Student(name : "Tony", age : 38, school : "香港大學") let Worker1 = Worker(name : "Tom", age : 18, factory : "鋼廠") let Worker2 = Worker(name : "Ben", age : 20, factory : "電廠") let people1: [Person] = [Student1, Student2, Student3, Worker1, Worker2] ① let people2: [AnyObject] = [Student1, Student2, Student3, Worker1, Worker2] ② let people3: [Any] = [Student1, Student2, Student3, Worker1, Worker2] ③ for item in people3 { ④ if let Student = item as? Student { println("Student school: \(Student.school)") } else if let Worker = item as? Worker { println("Worker factory: \(Worker.factory)") } }
上述代碼第①行是將5個實例放入Person數組中,第②行代碼是將5個實例放入AnyObject數組中,第③行代碼是將5個實例放入Any數組中。
這3種類型的數組都可以成功放入5個實例,而且可以在第④行使用for Int循環遍曆出來,其他的類型代碼不再解釋。
更多內容請關注國內第一本Swift圖書《Swift開發指南》
本書交流討論網站:https://www.51work6.com/swift.php
歡迎加入Swift技術討論群:362298485
歡迎關注智捷iOS課堂微信公共平台

最後更新:2017-04-03 05:40:12