泛型從字面來理解就是一種泛指的類型洪鸭,通過在類、方法仑扑、接口等指定泛型的類型览爵,可以避免重寫,達到重用的目的镇饮,而且可以通過泛型限制某些變量的類型蜓竹,本文通過幾個例子來闡述泛型在實際應(yīng)用開發(fā)中的作用,并從以下幾個方面介紹:
<h1>(1)泛型定義 </h1>
<h1>(2)泛型方法 </h1>
<h1>(3)泛型接口 </h1>
<h1>(4)泛型類 </h1>
<h2> 一储藐、泛型定義 </h2>
通過尖括號指定泛型類型<T>俱济,其中的T就是將要表示的類型被稱為占位符,swift的泛型可以指定任意類型钙勃,Int,String等蛛碌。
<h2> 二、泛型方法 </h2>
下面通過一個程序來介紹泛型方法肺缕,該方法用于交換兩個相同類型的值左医。
1.交換兩個Int類型的值
func swapTest(inout a : Int , inout b: Int) {
let c: Int = a
a = b
b = c
}
var a = 10
var b = 20
swapTest(&a, b: &b)
2.交換兩個String類型的值
func swapTest(inout a : String , inout b: String) {
let c: String = a
a = b
b = c
}
var a = "Tom"
var b = "Jerry"
swapTest(&a, b: &b)
此時就會發(fā)現(xiàn)問題,當(dāng)每次新增一種類型時同木,就需要再寫一個功能相同的函數(shù)浮梢,就會出現(xiàn)冗余,為了解決相同功能的函數(shù)在不同類型的對象間代碼公用彤路,那么我們需要使用泛型函數(shù):
func swapTest<T>(inout a : T , inout b: T) {
let c: T = a
a = b
b = c
}
var a = 10
var b = 20
swapTest(&a, b: &b)
var aString = "Tom"
var bString = "Jerry"
swapTest(&aString , &bString)
通過指定占位符來實現(xiàn)泛型函數(shù)秕硝,其中的T就為占位符,也就是說洲尊,函數(shù)在聲明的時候并沒有制定類型T远豺,該類型需要在調(diào)用的時候指定,再看看此時函數(shù)的參數(shù)坞嘀,也是通過占位符來限制a躯护、b的類型相同。當(dāng)調(diào)用函數(shù)swapTest時丽涩,函數(shù)會根據(jù)傳入?yún)?shù)的類型確定T的實際類型棺滞。
當(dāng)然泛型的占位符可以根據(jù)需要指定多個,例如:
func tuple<T , V>(a: T , b: V) -> (T , V) {
let tu: (T , V) = (a , b)
return tu
}
此方法根據(jù)指定兩個泛型占位符矢渊,為調(diào)用處返回一個元胞數(shù)組继准。需要注意的是,此處雖然指定了兩個泛型占位符矮男,但是不代表T與V的類型一定不相同移必,可以進行如下調(diào)用tuple(10,20)。
<h2> 三毡鉴、泛型接口(協(xié)議)</h2>
泛型接口與泛型方法和泛型類的定義略有不同崔泵,泛型接口在定義的時候并不直接通過尖括號聲明秒赤,而是在接口內(nèi)部通過associatedtype 并跟一個泛型占位符指定。如下所示
protocol CallBack {
associatedtype Element
var callBack: ((callData: Element!) -> Void)? { set get }
func load(d: Element!)
}
此時Element就是一個占位符憎瘸,泛型接口與普通的接口不一樣倒脓,不可以用作變量的類型,只能用于對泛型進行約束含思,例如說不可以存在如下代碼 var c = CallBack<Person>()。實現(xiàn)了該接口的類在進行l(wèi)oad后會調(diào)用callBack將數(shù)據(jù)傳回調(diào)用出甘晤。
<h2> 四含潘、泛型類 </h2>
泛型類的定義如下所示:
class PrjObj: NSObject {
override init() { super.init() }
func test() { }
}
class Test<T: PrjObj>: NSObject{
var testObj: [T]!
init(testObj: [T]) {
super.init()
self.testObj = testObj
}
func testObjFunc() {
for one in self.testObj {
one.test()
}
}
}
通過聲明Test類,該類的泛型占位符只接受PrjObj的子類线婚,用于進行項目相關(guān)類的測試遏弱。只要是PrjObj的子類,就可以加入到數(shù)組中塞弊,進行測試漱逸。
<h2> 泛型的實際應(yīng)用:</h2>
下面讓我來模擬一個實際應(yīng)用的場景,當(dāng)網(wǎng)絡(luò)請求成功后游沿,將數(shù)據(jù)回調(diào)(僅僅是模擬)饰抒。
首先需要一個回調(diào)接口
protocol CallBack {
associatedtype Element
var callBack: ((callData: Element!) -> Void)? { set get }
func load(d: Element!)
}
就像協(xié)議泛型所描述的,該協(xié)議的作用是限制實現(xiàn)了協(xié)議的類的功能诀黍,當(dāng)位于服務(wù)層的網(wǎng)絡(luò)請求響應(yīng)成功后袋坑,會調(diào)用load方法,load方法的作用是調(diào)用callBack眯勾,將數(shù)據(jù)回溯至相關(guān)頁面枣宫。
class Implement<T>: NSObject , CallBack {
typealias Element = T
var data: T!
var callBack: ((callData: T!) -> Void)?
func load(d: T!) {
self.data = d
self.callBack?(callData: self.data)
}
}
Implement類為實現(xiàn)了CallBack接口的泛型類,并實現(xiàn)了load方法吃环,load方法將設(shè)置data值也颤,并將調(diào)用callBack。
class CarListener: Implement{
static let instance: CarListener = CarListener()
}
class PersonListener: Implement{
static let instance: PersonListener = PersonListener()
}
聲明兩個Lisener的實現(xiàn)類郁轻,用來發(fā)送和接受相關(guān)類型的回調(diào)翅娶,PersonListener只負(fù)責(zé)用于Person類的回調(diào),CarListener只負(fù)責(zé)Car類的回調(diào)
class DispatchReceive {
var personListener: PersonListener = PersonListener.instance
var carListener: CarListener = CarListener.instance
func onMessage(person: Person!) {
}
func onMessage(car: Car!) -> Void {
}
init() {
self.personListener.callBack = onMessage
self.carListener.callBack = onMessage
}
}
DispatchReceive定義PersonListener 和 CarListener具體的回調(diào)方法范咨,當(dāng)相關(guān)的數(shù)據(jù)從服務(wù)器返回故觅,會調(diào)用各自的onMessage方法,傳入不同的參數(shù)渠啊。
class DispatchPost {
// data from server
func receivePersonFromServer() {
PersonListener.instance.load(Person())
}
func receiveCarFromServer() {
CarListener.instance.load(Car())
}
}
DispatchPost 服務(wù)器返回數(shù)據(jù)输吏,通過調(diào)用load方法將消息分發(fā)出去。