協(xié)議規(guī)定了用來實(shí)現(xiàn)某一特定功能所必需的方法和屬性。
任意能夠滿足協(xié)議要求的類型被稱為遵循(conform)這個(gè)協(xié)議。
類,結(jié)構(gòu)體或枚舉類型都可以遵循協(xié)議,并提供具體實(shí)現(xiàn)來完成協(xié)議定義的方法和功能。
引入
協(xié)議定義了一個(gè)大致的框架,規(guī)定里面有哪些屬性和方法,至于這個(gè)屬性的值是多少或者方法怎么實(shí)現(xiàn),我不管,留給別人實(shí)現(xiàn)?!駄ava里接口。
協(xié)議的定義
protocol SomeProtocol {
// 屬性
// 方法
}
屬性定義要求
1、不能用默認(rèn)值
2、必須指明類型: { set get } 來表示屬性是可讀可寫的,可讀屬性則用 { get }
3、必須用var修飾方法定義要求
1、方法不能有方法體
2、方法不可以有默認(rèn)值
//【協(xié)議的定義】——錯(cuò)誤示范
protocol Pet{
var name:String = "tets"
var age:Int = 18
func feed(food:String){}
}
/【協(xié)議的定義】——正確示范
protocol Pet{
var name:String{get set}
var age:Int{get}
func feed(food:String)
func shout(sound:String)
}
協(xié)議的遵循
- 語法
class 類:父類,協(xié)議1,協(xié)議2{
}
含義:類繼承某個(gè)父類,同時(shí)我遵守協(xié)議1和2
實(shí)現(xiàn)協(xié)議中的屬性
1、屬性可以設(shè)置默認(rèn)值
2、協(xié)議中屬性為可讀可寫,必須聲明為var
3、協(xié)議中屬性為可讀的,實(shí)現(xiàn)中可以聲明let或者var實(shí)現(xiàn)協(xié)議中的方法
1、方法中的參數(shù)可以傳默認(rèn)值
//協(xié)議的定義
protocol Pet{
var name:String{get set}
var age:Int{get}
func feed(food:String)
func shout(sound:String)
}
//【協(xié)議的遵守】
class Cat:Pet{
//屬性可以設(shè)置默認(rèn)值
//可讀可寫用var
var name: String = "mimi"
//只讀可以用let
let age: Int = 10
func feed(food: String) {
print("feed(food)")
}
//方法參數(shù)可以有默認(rèn)值
func shout(sound: String = "miaomiao") {
print("shout(sound)")
}
}
協(xié)議的繼承
協(xié)議能夠繼承一個(gè)或多個(gè)其他協(xié)議,可以在繼承的協(xié)議基礎(chǔ)上增加新的內(nèi)容要求。
協(xié)議的繼承語法與類的繼承相似,多個(gè)被繼承的協(xié)議間用逗號(hào)分隔:
//【協(xié)議的繼承】
protocol Protocol1{
func playBasketball()
func playFootball()
}
//讓第二個(gè)協(xié)議去繼承第一個(gè)協(xié)議,相當(dāng)于3個(gè)協(xié)議
protocol Protocol2:Protocol1{
func studyJava()
}
//【提問】 實(shí)現(xiàn)幾個(gè)方法? 實(shí)現(xiàn)協(xié)議中的所有方法
class Student:Protocol2{
func studyJava() {
<#code#>
}
func playBasketball() {
<#code#>
}
func playFootball() {
<#code#>
}
}
檢驗(yàn)協(xié)議的一致性
你可以使用is和as操作符來檢查是否遵循某一協(xié)議或強(qiáng)制轉(zhuǎn)化為某一類型。
- is操作符用來檢查實(shí)例是否遵循了某個(gè)協(xié)議。
- as?返回一個(gè)可選值,當(dāng)實(shí)例遵循協(xié)議時(shí),返回該協(xié)議類型;否則返回nil。
- as用以強(qiáng)制向下轉(zhuǎn)型,如果強(qiáng)轉(zhuǎn)失敗,會(huì)引起運(yùn)行時(shí)錯(cuò)誤。
協(xié)議中方法可選
如何實(shí)現(xiàn)讓協(xié)議中有些方法有些可以有選擇性的實(shí)現(xiàn)?
【方案一】 optional
在方法前添加@objc optional;在協(xié)議前用@objc修飾——不建議,建議用另一種方式,后面學(xué)
注意:@objc實(shí)在foundation框架中,如果注釋掉import 會(huì)報(bào)錯(cuò)
@objc protocol Protocol1{
@objc optional func playBasketball()
func playFootball()
}
class Student:Protocol1{
func playFootball() {
<#code#>
}
}
<meta charset="utf-8">
<article class="_2rhmJa">
協(xié)議的運(yùn)用
iOS開后后面經(jīng)常會(huì)用到,代理模式的基礎(chǔ)就是協(xié)議。
現(xiàn)實(shí)生活中代理模式很常見,比如我們想干一件事,但是這件事不是我們的專長(zhǎng),所以會(huì)委托別人來幫忙。
比如租房子,你不會(huì)在小區(qū)里挨個(gè)問誰有房子出租,我們回去找中介。
A-----目標(biāo) 搞不定
A---B->完成目標(biāo), A委托B,B就是我們說的代理。
注意: 代理不是想當(dāng)就當(dāng)?shù)?,需要滿足某些條件。
protocol BuyTicketProtocol{//凡是實(shí)現(xiàn)這個(gè)協(xié)議的人,我都可以讓他幫我買票
func buyTicket()
}
class MiShu:BuyTicketProtocol{
func buyTicket() {
print("秘書買票")
}
}
class HuangNiu:BuyTicketProtocol{
func buyTicket() {
print("黃牛買票")
}
}
class Boss{
var delegate:BuyTicketProtocol //找一個(gè)能買票的人,【思考】應(yīng)該是什么類型?
init(delegate:BuyTicketProtocol) {
self.delegate = delegate
}
func gotoBJ(){
//買票
delegate.buyTicket()
print("票到手,去出差")
}
}
var boss = Boss(delegate: HuangNiu())
boss.gotoBJ()
boss = Boss(delegate:MiShu())
boss.gotoBJ()
</article>