泛型(Generics)

泛型是Swift強(qiáng)大特征中的一個(gè),許多Swift 的標(biāo)準(zhǔn)庫是通過泛型來構(gòu)建的塔猾。Swift 中的數(shù)組和字典就是最好的體現(xiàn)环础。[type] type可以是任何的類型。

1闻伶、泛型解決的問題:

當(dāng)一個(gè)函數(shù)的功能相同滨攻,唯一的區(qū)別是傳入的參數(shù)的類型不同。泛型就是一個(gè)很好的特性蓝翰。將參數(shù)變?yōu)橐粋€(gè)泛型類型的變量光绕,就可以很好的解決這個(gè)問題。(所有的參數(shù)類型都用泛型類型來代替畜份,從而提高了代碼的復(fù)用)

2. ** 泛型函數(shù) **

// 兩個(gè)值的交換
// int 值
func swapTwoInt( a:  inout Int, b: inout Int) {
    let temporaryA = a
    a = b
    b = temporaryA
}

var a = 100
var b = 50
swapTwoInt(a: &a, b: &b)

print("a: \(a)")
print("b: \(b)")

a: 50
b: 100


// 字符串
func swapTwoString(a: inout String, b: inout String) {
    let temporaryA = a
    a = b
    b = temporaryA
}


var c = "zhangsan"
var d = "lisi"
swapTwoString(a: &c, b: &d)
print("c: \(c)")
print("d: \(d)")

c: lisi
d: zhangsan

// 使用 泛型 寫的交換函數(shù)
// 使用泛型函數(shù)可以交換任意類型的數(shù)據(jù),  只要 a, b 數(shù)據(jù)類型相同就可以
func swapToValue<T>(_ a: inout T, _ b: inout T) {
    let temp = a
    a = b
    b = temp
}

// 使用泛型函數(shù)進(jìn)行轉(zhuǎn)換
swapToValue(&a, &b)
swapToValue(&c, &d)
print("a: \(a)")
print("b: \(b)")
print("c: \(c)")
print("d: \(d)")


a: 50
b: 100
c: lisi
d: zhangsan

**T 的說明: **
函數(shù)的泛型版本中使用了 ** 占位類型名 **, 這里用 T 來表示 ( T : type 的第一個(gè)字母). 用來代替實(shí)際的類型名稱.

  • 在上述的函數(shù)中, T 沒有指明是什么類型. 但是指明了 a 和 b 必須為同一個(gè)類型.
    泛型函數(shù)的泛型類型只有在泛型函數(shù)調(diào)用的時(shí)候才可以知道泛型的類型是什么.
  • T 寫在函數(shù)的后面,并用尖括號括起來 <T> 表名 T 是在函數(shù)內(nèi)部定義的. swift 不會去查找 T 的實(shí)際類型.

swapToValue 的調(diào)用和平常的調(diào)用沒有區(qū)別, 只有一個(gè)條件, 參數(shù) a 和 b 的類型都是一樣的.
T 所代表的類型是通過函數(shù)的參數(shù)來推斷的.

類型參數(shù)
T 在函數(shù)中所表示的就是 類型參數(shù).
類型參數(shù)指定并命名一個(gè)占位類型, 寫在函數(shù)名的后面用 <> 括起來.如 <T>

可以提供多個(gè)類型參數(shù), 寫法是在尖括號中用 逗號 括起來.

<T, U, K>  // 這里代表提供三個(gè)類型的占位類型

命名類型參數(shù)
大寫字母開頭的駝峰命名法

Dictionary<Key, Value>
Array<Element>

Key , Value, Element 都是類型參數(shù)


常用的命名有 
T, U, V, K 等

類型約束
通過類型約束可以指定 類型 必須繼承某個(gè)類 或者遵守某個(gè)協(xié)議

Dictionary<Key, Value>

并不是所有的類型都可以做 key

// 完整的類型聲明
public struct Dictionary<Key : Hashable, Value> : Collection, ExpressibleByDictionaryLiteral { }

key 必須遵守 Hashable 協(xié)議.
只有遵守 Hashable 協(xié)議, 才可以保證 key 的唯一性.

類型約束語法

func someFunction<T: SomeClass, U: SomeProtocol>(someT: T, someU: U) {
    // 這里是泛型函數(shù)的函數(shù)體部分
}

關(guān)聯(lián)類型
關(guān)聯(lián)類型一般用在協(xié)議.
在協(xié)議中申明一個(gè)或多個(gè)關(guān)聯(lián)類型將會非常有用.

使用關(guān)鍵字 associatedtype

protocol Container {

    // 協(xié)議中聲明的關(guān)聯(lián)類型
    //  協(xié)議本身并不知道 ItemType 是什么類型
    associatedtype ItemType
    mutating func append(item: ItemType)
    var count: Int { get }
    subscript(i: Int) -> ItemType { get }
}
// Stack 遵守了容器協(xié)議
struct Stack<Element>: Container {
    // Stack<Element> 的原始實(shí)現(xiàn)部分
    var items = [Element]()
    mutating func push(item: Element) {
        items.append(item)
    }
    mutating func pop() -> Element {
        return items.removeLast()
    }

    // Container 協(xié)議的實(shí)現(xiàn)部分
    // 通過容器的實(shí)現(xiàn)來知道 關(guān)聯(lián)類型 的具體類型是什么類型
    // 這里來確認(rèn) 關(guān)聯(lián)類型是什么類型. 
    typealias ItemType = Element   // 刪除這一行,一樣的可以工作. 類型可以推斷出來

    mutating func append(item: Element) {
        self.push(item)
    }
    var count: Int {
        return items.count
    }
    subscript(i: Int) -> Element {
        return items[i]
    }
}


//  通過已經(jīng)存在的類型來指定關(guān)聯(lián)類型的真實(shí)類型
extension Array: Container {   }

where 子句
給類型參數(shù)添加強(qiáng)制性的約束

3. 泛型類型

Swift 允許定義泛型類型 .
自定義的泛型枚舉, 結(jié)構(gòu)體, 類可以用于任何類型.

struct IntStack {
    var items = [Int]()
    
    mutating func push(item: Int) {
        items.append(item)
    }
    
    mutating func pop() -> Int {
        return items.removeLast()
    }
}


var intStack = IntStack()
// 壓棧
intStack.push(item: 0)
intStack.push(item: 1)
intStack.push(item: 2)
intStack.push(item: 3)
intStack.push(item: 4)
intStack.push(item: 5)
intStack.push(item: 6)
intStack.push(item: 7)
intStack.push(item: 8)

// 彈棧
print(intStack)
intStack.pop()

print(intStack)
intStack.pop()

print(intStack)
intStack.pop()

print(intStack)
intStack.pop()

print(intStack)
intStack.pop()

print(intStack)
intStack.pop()

print(intStack)
intStack.pop()

print(intStack)
intStack.pop()

print(intStack)
intStack.pop()


// 泛型結(jié)構(gòu)體
struct Stack<Element>{
    var items = [Element]()
    
    mutating func push(item: Element) {
        items.append(item)
    }
    
    mutating func pop() -> Element {
        return items.removeLast()
    }
}

// 調(diào)用的時(shí)候要 指定數(shù)據(jù)類型
var tempStack = Stack<Int>()
tempStack.push(item: 0)
tempStack.push(item: 1)
tempStack.push(item: 2)
tempStack.push(item: 3)
tempStack.push(item: 4)
tempStack.push(item: 5)

print(tempStack)
tempStack.pop()

print(tempStack)
tempStack.pop()

print(tempStack)
tempStack.pop()

print(tempStack)
tempStack.pop()

print(tempStack)
tempStack.pop()

print(tempStack)
tempStack.pop()

// 調(diào)用的時(shí)候要 指定數(shù)據(jù)類型
var tempStack = Stack<Int>()

通過創(chuàng)建的時(shí)候指定類型. 這個(gè)結(jié)構(gòu)體可以用于任何類型.

擴(kuò)展泛型類型

// 泛型的擴(kuò)展
extension Stack {
    var topItem: Element? {
        return items.isEmpty ? nil : items[items.count - 1]
    }
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末诞帐,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子爆雹,更是在濱河造成了極大的恐慌停蕉,老刑警劉巖愕鼓,帶你破解...
    沈念sama閱讀 219,490評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異慧起,居然都是意外死亡菇晃,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評論 3 395
  • 文/潘曉璐 我一進(jìn)店門蚓挤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來磺送,“玉大人,你說我怎么就攤上這事灿意」啦樱” “怎么了?”我有些...
    開封第一講書人閱讀 165,830評論 0 356
  • 文/不壞的土叔 我叫張陵缤剧,是天一觀的道長甲捏。 經(jīng)常有香客問我,道長鞭执,這世上最難降的妖魔是什么司顿? 我笑而不...
    開封第一講書人閱讀 58,957評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮兄纺,結(jié)果婚禮上大溜,老公的妹妹穿的比我還像新娘。我一直安慰自己估脆,他們只是感情好钦奋,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評論 6 393
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著疙赠,像睡著了一般付材。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上圃阳,一...
    開封第一講書人閱讀 51,754評論 1 307
  • 那天厌衔,我揣著相機(jī)與錄音,去河邊找鬼捍岳。 笑死富寿,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的锣夹。 我是一名探鬼主播页徐,決...
    沈念sama閱讀 40,464評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼银萍!你這毒婦竟也來了变勇?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤贴唇,失蹤者是張志新(化名)和其女友劉穎搀绣,沒想到半個(gè)月后赃梧,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,847評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡豌熄,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了物咳。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片锣险。...
    茶點(diǎn)故事閱讀 40,137評論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖览闰,靈堂內(nèi)的尸體忽然破棺而出芯肤,到底是詐尸還是另有隱情,我是刑警寧澤压鉴,帶...
    沈念sama閱讀 35,819評論 5 346
  • 正文 年R本政府宣布崖咨,位于F島的核電站,受9級特大地震影響油吭,放射性物質(zhì)發(fā)生泄漏击蹲。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評論 3 331
  • 文/蒙蒙 一婉宰、第九天 我趴在偏房一處隱蔽的房頂上張望歌豺。 院中可真熱鬧,春花似錦心包、人聲如沸类咧。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽痕惋。三九已至,卻和暖如春娃殖,著一層夾襖步出監(jiān)牢的瞬間值戳,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評論 1 272
  • 我被黑心中介騙來泰國打工炉爆, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留述寡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,409評論 3 373
  • 正文 我出身青樓叶洞,卻偏偏與公主長得像鲫凶,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子衩辟,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評論 2 355

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

  • 泛型代碼能夠根據(jù)自定義的需求螟炫,編寫出適用于任意類型、靈活可重用的函數(shù)及類型艺晴。 泛型能夠避免代碼的重復(fù)昼钻,用一種清晰和...
    勇往直前888閱讀 386評論 0 0
  • 本章將會介紹 泛型所解決的問題泛型函數(shù)類型參數(shù)命名類型參數(shù)泛型類型擴(kuò)展一個(gè)泛型類型類型約束關(guān)聯(lián)類型泛型 Where...
    寒橋閱讀 639評論 0 2
  • 泛型(Generics) 泛型代碼允許你定義適用于任何類型的,符合你設(shè)置的要求的,靈活且可重用的 函數(shù)和類型掸屡。泛型...
    果啤閱讀 678評論 0 0
  • 【Swift 3.1】23 - 泛型 (Generics) 自從蘋果2014年發(fā)布Swift,到現(xiàn)在已經(jīng)兩年多了然评,...
    Lebron_James閱讀 1,087評論 2 0
  • 我還在癡癡等 等轉(zhuǎn)角時(shí)的擁抱 也許你也知道 有顆心為你慌亂的跳 也許你永遠(yuǎn)不會知道 我把它隱藏的太好 你對我太過于...
    不爭不搶不憂閱讀 306評論 7 17