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