標(biāo)簽(空格分隔): IOS-Swift
[toc]
為何使用泛型
在編程語(yǔ)言中,往往有很多的數(shù)據(jù)類(lèi)型,如果我們需要對(duì)每個(gè)類(lèi)型都進(jìn)行相同的操作,那么就需要為每種類(lèi)型都編寫(xiě)一段相同的代碼,比如我們想要比較兩個(gè)值得大小.一般會(huì)寫(xiě)成這樣
func maxInt(a:Int,b:Int) ->Int{
return a>b?a:b
}
這樣寫(xiě)是沒(méi)有問(wèn)題的,但是參數(shù)只能接受Int類(lèi)型的,如果再想比較float的值呢?那么就需要再寫(xiě)一個(gè)這樣的方法,但是如果再有其他的類(lèi)型的呢?這樣就會(huì)造成很多重復(fù)的代碼.而泛型就是為了解決這個(gè)問(wèn)題存在的.
基本的使用
為了解決上面的問(wèn)題,我們可以做這樣的事情,我們可以用一個(gè)任意的類(lèi)型,讓這個(gè)方法可以接受所有的類(lèi)型,于是代碼就變成了這樣
func maxAny<T>(a:T,b:T) ->T{
return a>b?a:b
}
這里的
T
是一個(gè)類(lèi)型占位符,由于沒(méi)有為T
設(shè)定協(xié)議,所以它表示任意類(lèi)型.讓函數(shù)的兩個(gè)參數(shù)都是T
類(lèi)型,這樣函數(shù)就可以接受任何類(lèi)型了.但是這行代碼會(huì)出現(xiàn)一個(gè)錯(cuò)誤:
Binary operator '>' cannot be applied to two 'T' operands
為啥呢? 因?yàn)?>
比較符只能作用于符合Comparable
協(xié)議的類(lèi)型,而任意類(lèi)型的T
不符合任何協(xié)議.這里先不管,我們使用輸出語(yǔ)句試試這個(gè)方法是否真的能接受任何數(shù)據(jù)類(lèi)型
func maxAny<T>(a:T,b:T) -> T {
print(a)
return b
}
//調(diào)用方法,分別傳String和Int的類(lèi)型
self.maxAny("a", b: "asd")
self.maxAny(12, b: 13)
//輸出結(jié)果
a
12
兩種類(lèi)型都輸出正常,表示這個(gè)方法確實(shí)可以接受兩種類(lèi)型
那么如何讓
a
和b
可以比較呢?只要為T
添加Comparable
協(xié)議即可
func maxAny<T:Comparable>(a:T,b:T) ->T{
return a>b?a:b
}
這樣就表示,只有符合
Comparable
協(xié)議的類(lèi)型才可以作為方法的參數(shù)傳遞進(jìn)來(lái).Comparable
協(xié)議是專(zhuān)門(mén)用來(lái)比較的協(xié)議,不支持比較的類(lèi)型不會(huì)添加這個(gè)協(xié)議,比如String
類(lèi)型.
print(self.maxAny(2, b: 3)) ;
print(self.maxAny(3.2, b: 4.2))
//分別調(diào)用方法傳遞Int類(lèi)型和double類(lèi)型的參數(shù)
//輸出結(jié)果
3
4.2
這里就完成了,只要一個(gè)方法,就可以傳遞多種類(lèi)型的參數(shù),這個(gè)就是使用泛型的主要目的
常見(jiàn)的泛型使用
swift中,Array是可以接受任何類(lèi)型的.但是我們可以使用泛型,讓Array只接受一種數(shù)據(jù)類(lèi)型.
var array:Array<Int> = Array()//聲明一個(gè)數(shù)組
array.append(10)//將整數(shù)加入數(shù)組
array.append("str")//將String加入數(shù)組
這時(shí)就會(huì)發(fā)現(xiàn),第三行代碼報(bào)錯(cuò)了,
Cannot convert value of type 'String' to expected argument type 'Int'
就是說(shuō)String
類(lèi)型無(wú)法轉(zhuǎn)換成預(yù)期中得Int
類(lèi)型,這說(shuō)明數(shù)組就只能存放Int
類(lèi)型的了.這樣有什么好處呢?這樣我們可以很方便的清楚數(shù)組中得類(lèi)型,使用起來(lái)更安全.除了數(shù)組,字典也同樣支持泛型.
在類(lèi)中使用泛型
模擬一個(gè)類(lèi)似于Array的類(lèi)
class Test<T> {
var items = [T]()
func append(item:T){
items.append(item)
}
}
//接下來(lái)是使用
var a:Test<Int> = Test()
a.append(12)
a.append("asd")
//這里第三行同樣會(huì)報(bào)錯(cuò),和Array類(lèi)型是一樣的
這樣做的好處就是這個(gè)類(lèi)本身可以接收任意的類(lèi)型,也可以指定類(lèi)接收的類(lèi)型.