Swift中的协议

1. 协议语法

protocol SomeProtocol {
    // code
}

要让自定义类型遵循某个协议,在类型名称后加上协议名称即可,中间以冒号(:)分隔,遵循多个协议时,各协议之间用逗号()分隔:

struct SomeStruct: FirstProtocol, AnotherProtocol {
    // code
}

拥有父类的类在遵循协议时,应该将父类名放在协议名之前,以逗号分隔:

class SomeClass: SomeSuperClass, FirstProtocol, AnotherProtocol {
    // code
}

2. 属性要求

protocol SomeProtocol {
    var mustBeSettable: Int { get set } //可读可写的变量属性
    var doesNotNeedToBeSettable: Int { get } //可读变量属性
}
protocol AnotherProtocol {
    static var someTypeProperty: Int { get set } //类型属性,还可以用class关键字声明
}

如下所示,这是一个只含有一个实例属性要求的协议。这个协议表示,任何遵循FullyNamed的类型,都必须有一个可读的String类型的实力属性fullName

protocol FullyNamed {
    var fullName: String { get }
}

下面是一个遵循FullyNamed协议的简单结构体:

struct Person: FullyNamed {
    var fullName: String
}

let john = Person(fullName: "John Higgins")

下面是一个更为复杂的类,它适配并遵循了FullyNamed协议:

class StarShip: FullyNamed {
    var prefix: String?
    var name: String
    init(name: String, prefix: String? = nil) {
        self.name = name
        self.prefix = prefix
    }
    var fullName: String {
        return (prefix != nil ? prefix! + " " : "") + name
    }
}

var ncc1701 = StarShip(name: "Enterprise", prefix: "USS")
print(ncc1701.fullName)
// print "USS Enterprise"

3. 方法要求

protocol SomeProtocol {
    static func someTypeMethod() // 类方法还可以用class关键字修饰
}

下面的例子定义了一个只含有一个实例方法的协议:

protocol RandomNumberGenerator {
    func random() -> Double
}

下面的类实现了一个叫做线性同余生成器的伪随机数算法:

class LinearCongruentialGenerator: RandomNumberGenerator {
    var lastRandom = 42.0
    let m = 139968.0
    let a = 3877.0
    let c = 29573.0
    func random() -> Double {
        lastRandom = (lastRandom * a + c).truncatingRemainder(dividingBy: m)
        return lastRandom / m
    }
}

4. Mutating方法要求

实现协议中的mutating方法时,若是类类型,则不用写mutating关键字,而对于结构体和枚举,则必须写mutating关键字。

protocol Togglable {
    mutating func toggle()
}

enum OnOffSwitch: Togglable {
    case Off, On
    mutating func toggle() {
        switch self {
        case .Off:
            self = .On
        case .On:
            self = .Off
        }
    }
}

var lightSwitch = OnOffSwitch.Off
lightSwitch.toggle() // On

5. 构造器要求

protocol SomeProtocol {
    init(someParameter: Int)
}

class SomeClass: SomeProtocol {
    required init(someParameter: Int) {
        // 构造器实现部分
    }
}

如果一个子类重写了父类的指定构造器,并且该构造器满足了某个协议的要求,那么该构造器的实现需要同时标注requiredoverride修饰符:

class SomeSuperClass {
    init() {
        // 构造器实现部分
    }
}

class SomeSubClass: SomeSuperClass, SomeProtocol {
    required override init() {
        // 构造器实现部分
    }
}

6. 协议作为类型

class Dice {
    let sides: Int
    let generator: RandomNumberGenerator
    init(sides: Int, generotor: RandomNumberGenerator) {
        self.sides = sides
        self.generator = generator
    }
    func roll() -> Int {
        return Int(generator.random() * Double(sides)) + 1
    }
}

generator属性的类型为RandomNumberGenerator,因此任何遵循了RandomNumberGenerator协议的类型的实例都可以赋值给generator,除此之外并无其他要求。

7. 委托(代理)模式

protocol SayHelloDelegate {
    func sayHello(name: String)
}

class ClassA {
    var delegate: SayHelloDelegate?
    var name = "Lucy"
    func play() {
        delegate?.sayHello(name: name)
    }
}

class ClassB: SayHelloDelegate {
    var name = "Lily"
    func sayHello(name: String) {
        print("\(name) 请 \(self.name) 帮她 say Hello")
    }
}

var ca = ClassA()
var cb = ClassB()
ca.delegate = cb
ca.play()
// print "Lucy 请 Lily 帮她 say Hello"

8. 协议中添加扩展

protocol Score {
    var math: Int { get set }
    var english: Int { get set }
    func mathPercent() -> Double
}

struct Puple: Score {
    var math: Int
    var english: Int
    func mathPercent() -> Double {
        return Double(math) / Double(math + english)
    }
}

let p1 = Puple(math: 90, english: 80)
p1.mathPercent() // 0.529

extension Score {
    func mathPercent() -> Double {
        return Double(math) / Double(math + english)
    }
}

struct CollageStudent: Score {
    var math: Int
    var english: Int
}

let c1 = CollageStudent(math: 80, english: 80)
c1.mathPercent() // 0.5

如此,我们可以不实现mathPercent方法也能计算出数学所占分数的比例。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容