精通Swift(15)- 協(xié)議

協(xié)議(Protocol)

????協(xié)議可以用來定義方法乾蓬、屬性施戴、下標(biāo)的聲明纸俭,協(xié)議可以被枚舉皇耗、結(jié)構(gòu)體、類遵守(多個(gè)協(xié)議之間用逗號(hào)隔開)

protocol Drawable {

? ? func draw()

? ? var x: Int { get set }

? ? var y: Int { get }

? ? subscript(index: Int) -> Int { get set }

}

protocol Test1 {}

protocol Test2 {}

protocol Test3 {}

class TestClass : Test1, Test2, Test3 {}

????協(xié)議中定義方法時(shí)不能有默認(rèn)參數(shù)值

????默認(rèn)情況下揍很,協(xié)議中定義的內(nèi)容必須全部都實(shí)現(xiàn)?

????也有辦法辦到只實(shí)現(xiàn)部分內(nèi)容郎楼,以后會(huì)講到

協(xié)議中的屬性

protocol Drawable {? ? func draw()

????var?x:?Int?{?get set?}

????var?y:?Int?{?get?}

????subscript(index:?Int) ->?Int?{?get set?}

}

協(xié)議中定義屬性時(shí)必須用var關(guān)鍵字

實(shí)現(xiàn)協(xié)議時(shí)的屬性權(quán)限要不小于協(xié)議中定義的屬性權(quán)限

協(xié)議定義get万伤、set,用var存儲(chǔ)屬性或get呜袁、set計(jì)算屬性去實(shí)現(xiàn)?

協(xié)議定義get敌买,用任何屬性都可以實(shí)現(xiàn)

class Person : Drawable {? ? var x: Int = 0

? ? let y: Int = 0? ? func draw() {

? ? ? ? print("Person draw")

? ? }

????subscript(index:?Int) ->?Int?{?set?{}

????get?{ index } }

}

class Person : Drawable {? ??

????var x: Int {

????????get?{?0?}

????????set?{} }

????var?y:?Int?{?0?}

????func?draw() {?print("Person draw") }?

????subscript(index:?Int) ->?Int?{

????????set?{}

????????get?{ index } }

}

static、class

????為了保證通用阶界,協(xié)議中必須用static定義類型方法虹钮、類型屬性、類型下標(biāo)

protocol Drawable {

? ? static func draw()

}

class Person1 : Drawable {

? ? class func draw() {

? ? ? ? print("Person1 draw")

? ? }

}

class Person2 : Drawable {

? ? static func draw() {

? ? ? ? print("Person2 draw")

? ? }

}

mutating

????只有將協(xié)議中的實(shí)例方法標(biāo)記為mutating才允許結(jié)構(gòu)體膘融、枚舉的具體實(shí)現(xiàn)修改自身內(nèi)存

????類在實(shí)現(xiàn)方法時(shí)不用加mutating芙粱,枚舉、結(jié)構(gòu)體才需要加mutating

protocol Drawable {

? ? mutating func draw()

}

class Size : Drawable {

? ? var width: Int = 0

? ? func draw() {

width = 10 }

}

struct Point : Drawable {

? ? var x: Int = 0

? ? mutating func draw() {

? ? ? ? x = 10

} }

init

????協(xié)議中還可以定義初始化器init?

????非final類實(shí)現(xiàn)時(shí)必須加上required?

protocol Drawable {

? ? init(x: Int, y: Int)

}

class Point : Drawable {

? ? required init(x: Int, y: Int) {}

}

final class Size : Drawable {

? ? init(x: Int, y: Int) {}

}

????如果從協(xié)議實(shí)現(xiàn)的初始化器托启,剛好是重寫了父類的指定初始化器?

????那么這個(gè)初始化必須同時(shí)加required宅倒、override ? ?

protocol Livable {

? ? init(age: Int)

}

class Person {

? ? init(age: Int) {}

}

class Student : Person, Livable {

? ? required override init(age: Int) {

? ? ? ? super.init(age: age)

? ? }

}

init、init?屯耸、init!

????協(xié)議中定義的init?拐迁、init!,可以用init疗绣、init?线召、init!去實(shí)現(xiàn)?

????協(xié)議中定義的init,可以用init多矮、init!去實(shí)現(xiàn)

? protocol Livable {

? ? init()

? ? init?(age: Int)

? ? init!(no: Int)

}

class Person : Livable {

? ? required init() {}

? ? // required init!() {}

? ? required init?(age: Int) {}

? ? // required init!(age: Int) {}

? ? // required init(age: Int) {}

? ? required init!(no: Int) {}

? ? // required init?(no: Int) {}

? ? // required init(no: Int) {}

}

協(xié)議的繼承

????一個(gè)協(xié)議可以繼承其他協(xié)議

protocol Runnable {

? ? func run()

}

protocol Livable : Runnable {

? ? func breath()

}

class Person : Livable {

? ? func breath() {}

? ? func run() {}

}

協(xié)議組合

protocol Livable {}

protocol Runnable {}

class Person {}

協(xié)議組合缓淹,可以包含1個(gè)類類型(最多1個(gè))

// 接收Person或者其子類的實(shí)例

func fn0(obj: Person) {}

// 接收遵守Livable協(xié)議的實(shí)例

func fn1(obj: Livable) {}

// 接收同時(shí)遵守Livable、Runnable協(xié)議的實(shí)例

func fn2(obj: Livable & Runnable) {}

// 接收同時(shí)遵守Livable塔逃、Runnable協(xié)議讯壶、并且是Person或者其子類的實(shí)例 func fn3(obj: Person & Livable & Runnable) {}

typealias RealPerson = Person & Livable & Runnable

// 接收同時(shí)遵守Livable、Runnable協(xié)議湾盗、并且是Person或者其子類的實(shí)例 func fn4(obj: RealPerson) {}

CaseIterable

????讓枚舉遵守CaseIterable協(xié)議伏蚊,可以實(shí)現(xiàn)遍歷枚舉值

enum Season : CaseIterable {

? ? case spring, summer, autumn, winter

}

let seasons = Season.allCases

print(seasons.count) // 4

for season in seasons {

? ? print(season)

} // spring summer autumn winter

CustomStringConvertible

????遵守CustomStringConvertible、 CustomDebugStringConvertible協(xié)議格粪,都可以自定義實(shí)例的打印字符串

class Person : CustomStringConvertible, CustomDebugStringConvertible {?

????var age = 0

????var description: String { "person_\(age)" }

????var debugDescription: String { "debug_person_\(age)" } }

????var person = Person()

????print(person) // person_0?

????debugPrint(person) // debug_person_0

print調(diào)用的是CustomStringConvertible協(xié)議的description

debugPrint躏吊、po調(diào)用的是CustomDebugStringConvertible協(xié)議的debugDescription

Any、AnyObject

????Swift提供了2種特殊的類型:Any帐萎、AnyObject?

????????Any:可以代表任意類型(枚舉比伏、結(jié)構(gòu)體、類疆导,也包括函數(shù)類型)?

????????AnyObject:可以代表任意類類型(在協(xié)議后面寫上: AnyObject代表只有類能遵守這個(gè)協(xié)議)?

????在協(xié)議后面寫上: class也代表只有類能遵守這個(gè)協(xié)議

var stu: Any = 10

stu = "Jack"

stu = Student()

// 創(chuàng)建1個(gè)能存放任意類型的數(shù)組 //?

var data = Array<Any>()?

var data = [Any]()?

data.append(1)?

data.append(3.14)?

data.append(Student())?

data.append("Jack")?

data.append({ 10 })

is赁项、as?、as!、as

????is用來判斷是否為某種類型肤舞,as用來做強(qiáng)制類型轉(zhuǎn)換

protocol Runnable { func run() }?

class Person {}

class Student : Person, Runnable {

? ? func run() {

? ? ? ? print("Student run")

? ? }

? ? func study() {

? ? ? ? print("Student study")

? ? }

}

var stu: Any = 10

(stu as? Student)?.study() // 沒有調(diào)用study stu = Student()

(stu as? Student)?.study() // Student study (stu as! Student).study() // Student study (stu as? Runnable)?.run() // Student run

var data = [Any]() data.append(Int("123") as Any)

var d = 10 as Double

print(d) // 10.0

var stu: Any = 10

print(stu is Int) // true

stu = "Jack"

print(stu is String) // true stu = Student()

print(stu is Person) // true print(stu is Student) // true print(stu is Runnable) // true

X.self紫新、X.Type、AnyClass

????X.self是一個(gè)元類型(metadata)的指針李剖,metadata存放著類型相關(guān)信息 n X.self屬于X.Type類型

class Person {}

class Student : Person {}

var perType: Person.Type = Person.self var stuType: Student.Type = Student.self perType = Student.self

----------

var anyType: AnyObject.Type = Person.self?

anyType = Student.self

public typealias AnyClass = AnyObject.Type?

var anyType2: AnyClass = Person.self?

anyType2 = Student.self

---------

var per = Person()

var perType = type(of: per) // Person.self print(Person.self == type(of: per)) // true

元類型的應(yīng)用

class Animal { required init() {} } class Cat : Animal {}

class Dog : Animal {}

class Pig : Animal {}

func create(_ clses: [Animal.Type]) -> [Animal] {?

????var arr = [Animal]()

? ? for cls in clses {

????????arr.append(cls.init()) }

????????return arr }

print(create([Cat.self, Dog.self, Pig.self]))

--------------------------

import Foundation

class Person {

? ? var age: Int = 0

}

class Student : Person {

? ? var no: Int = 0

}

print(class_getInstanceSize(Student.self)) // 32 print(class_getSuperclass(Student.self)!) // Person print(class_getSuperclass(Person.self)!) // Swift._SwiftObject

從結(jié)果可以看得出來芒率,Swift還有個(gè)隱藏的基類:Swift._SwiftObject?

可以參考Swift源碼:https://github.com/apple/swift/blob/master/stdlib/public/runtime/SwiftObject.h

Self

Self代表當(dāng)前類型

class Person {

? ? var age = 1

? ? static var count = 2

? ? func run() {

? ? ? ? print(self.age) // 1

? ? ? ? print(Self.count) // 2

? ? }

}

Self一般用作返回值類型,限定返回值跟方法調(diào)用者必須是同一類型(也可以作為參數(shù)類型)

? protocol Runnable {

? ? func test() -> Self

}

class Person : Runnable {

? ? required init() {}

????func test() -> Self { type(of: self).init() }?

}

class Student : Person {}

var p = Person()

// Person

print(p.test())

var stu = Student()

// Student

print(stu.test())

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末篙顺,一起剝皮案震驚了整個(gè)濱河市偶芍,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌德玫,老刑警劉巖匪蟀,帶你破解...
    沈念sama閱讀 218,546評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異宰僧,居然都是意外死亡材彪,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,224評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門琴儿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來段化,“玉大人,你說我怎么就攤上這事造成∠匝” “怎么了?”我有些...
    開封第一講書人閱讀 164,911評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵晒屎,是天一觀的道長喘蟆。 經(jīng)常有香客問我,道長鼓鲁,這世上最難降的妖魔是什么蕴轨? 我笑而不...
    開封第一講書人閱讀 58,737評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮骇吭,結(jié)果婚禮上橙弱,老公的妹妹穿的比我還像新娘。我一直安慰自己绵跷,他們只是感情好膘螟,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,753評(píng)論 6 392
  • 文/花漫 我一把揭開白布成福。 她就那樣靜靜地躺著碾局,像睡著了一般。 火紅的嫁衣襯著肌膚如雪奴艾。 梳的紋絲不亂的頭發(fā)上净当,一...
    開封第一講書人閱讀 51,598評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼像啼。 笑死俘闯,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的忽冻。 我是一名探鬼主播真朗,決...
    沈念sama閱讀 40,338評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼僧诚!你這毒婦竟也來了遮婶?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,249評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤湖笨,失蹤者是張志新(化名)和其女友劉穎旗扑,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體慈省,經(jīng)...
    沈念sama閱讀 45,696評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡臀防,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,888評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了边败。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片袱衷。...
    茶點(diǎn)故事閱讀 40,013評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖放闺,靈堂內(nèi)的尸體忽然破棺而出祟昭,到底是詐尸還是另有隱情,我是刑警寧澤怖侦,帶...
    沈念sama閱讀 35,731評(píng)論 5 346
  • 正文 年R本政府宣布篡悟,位于F島的核電站,受9級(jí)特大地震影響匾寝,放射性物質(zhì)發(fā)生泄漏搬葬。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,348評(píng)論 3 330
  • 文/蒙蒙 一艳悔、第九天 我趴在偏房一處隱蔽的房頂上張望急凰。 院中可真熱鬧,春花似錦猜年、人聲如沸抡锈。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,929評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽床三。三九已至,卻和暖如春杨幼,著一層夾襖步出監(jiān)牢的瞬間撇簿,已是汗流浹背聂渊。 一陣腳步聲響...
    開封第一講書人閱讀 33,048評(píng)論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留四瘫,地道東北人汉嗽。 一個(gè)月前我還...
    沈念sama閱讀 48,203評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像找蜜,于是被迫代替她去往敵國和親饼暑。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,960評(píng)論 2 355

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