iOS学习笔记45-Swift(五)协议

一、Swift协议

协议是为方法、属性等定义一套规范,没有具体的实现,类似于Java中的抽象接口,它只是描述了方法或属性的骨架,而不是实现。方法和属性实现还需要通过定义类,函数和枚举完成。

1. 协议定义

//协议定义通过关键字protocol
protocol SomeProtocol {
    //协议定义
}
//协议可以继承一个或者多个协议
protocol SomeProtocol2: SomeProtocol {
    //协议定义
}
//结构体实现协议
struct SomeStructure: SomeProtocol, SomeProtocol2 {
    //结构体定义
}
//类实现协议和继承父类,协议一般都写在父类后面
class SomeSuperclass {
    //父类定义
}
class SomeClass: SomeSuperclass, SomeProtocol, SomeProtocol2 {
    //子类定义
}

2. 属性要求

协议不指定是否该属性应该是一个存储属性或者计算属性,它只指定所需的属性名称和读写类型。属性要求总是声明为变量属性,用var关键字做前缀。

protocol ClassProtocol {
   static var present: Bool { get set }  //要求该属性可读可写,并且是静态的
   var subject: String { get }    //要求该属性可读
   var stname: String { get set }   //要求该属性可读可写
}
//定义类来实现协议
class MyClass: ClassProtocol {
   static var present = false  //如果没有实现协议的属性要求,会直接报错
   var subject = "Swift Protocols" //该属性设置为可读可写,也是满足协议要求的
   var stname = "Class"
   func attendance() -> String {
      return "The \(self.stname) has secured 99% attendance"
   }
   func markssecured() -> String {
      return "\(self.stname) has \(self.subject)"
   }
}
//创建对象
var classa = MyClass()
print(classa.attendance()) //结果:The Class has secured 99% attendance
print(classa.markssecured()) //结果:Class has Swift Protocols

3. 普通实例方法要求

协议可以要求指定实例方法和类型方法被一致的类型实现。这些方法被写为协议定义的一部分,跟普通实例和类型方法完全一样,但是没有大括号或方法体。可变参数是允许的,普通方法也遵循同样的规则,不过不允许给协议方法参数指定默认值。

//定义协议,指定方法要求
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) % m)
        return lastRandom / m
    }
}
let generator = LinearCongruentialGenerator()
print("随机数: \(generator.random())") //结果:随机数: 0.37464991998171
print("另一个随机数: \(generator.random())") //结果:另一个随机数: 0.729023776863283

4. Mutating方法要求

有时需要一个方法来修改它属于的实例。对值类型实例方法(即结构和枚举),你将mutating关键字放在方法func关键字之前,表明该方法允许修改所属实例的任何属性。这个过程描述在实例方法内修改值类型,通常用于结构体和枚举。

protocol Togglable {
    //协议的Mutating方法要求,允许在该方法内修改值类型
    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() //此时lightSwitch变成了OnOffSwitch.On
switch(lightSwitch) {
    case .On:
        print("开关On")
    case .Off:
        print("开关Off")
}
//打印:开关On

5. 初始化构造器要求

协议SomeProtocol中不光可以声明方法/属性/下标,还可以声明构造器,但在Swift中,除了某些特殊情况外,构造器是不被子类继承的,所以SomeClass中虽然能够保证定义了协议要求的构造器,但不能保证SomeClass的子类中也定义了协议要求的构造器。所以我们需要在实现协议要求的构造器时,使用required关键字确保SomeClass的子类必须也得实现这个构造器。

protocol TcpProtocol {
    //初始化构造器要求
    init(aprot: Int)
}
class TcpClass: TcpProtocol {
    var aprot: Int
    //实现协议的初始化要求时,必须使用required关键字确保子类必须也得实现这个构造器
    required init(aprot: Int) {
        self.aprot = aprot
    }
}
var tcp = TcpClass(aprot: 20)
print(tcp.aprot)

6. 协议类型使用

协议可以作为类型访问:
  • 函数,方法或初始化作为一个参数或返回类型

  • 常量,变量或属性

  • 数组,字典或其他容器作为项目

//定义随机数生成器协议
protocol RandomNumberGenerator {
    func random() -> Double
}
//实现RandomNumberGenerator协议的类
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) % m)
        return lastRandom / m
    }
}
//定义骰子类
class Dice {
    let sides: Int //表示「骰子」有几个面
    let generator: RandomNumberGenerator //随机数生成器
    
    //指定构造器,RandomNumberGenerator是一个协议名
    init(sides: Int, generator: RandomNumberGenerator) {
        self.sides = sides
        self.generator = generator
    }
    
    //摇动「骰子」
    func roll() -> Int {
        return Int(generator.random() * Double(sides)) + 1
    }
}
//创建一个6面骰子
var dice6 = Dice(sides: 6, generator: LinearCongruentialGenerator())
for i in 1...5 {
    print("摇动骰子:\(dice6.roll())")
}
/*
摇动骰子:3
摇动骰子:5
摇动骰子:4
摇动骰子:5
摇动骰子:4
*/

7. 协议组合

协议组合对于要求一个类型立即符合多种协议是有用的。

protocol Named {
    var name: String { get }
}
protocol Aged {
    var age: Int { get }
}
//定义结构体实现上面2个协议
struct Person: Named, Aged {
    var name: String
    var age: Int
}
//定义一个函数,接受一个符合Named和Aged协议的类型
func wishHappyBirthday(celebrator: protocol<Named, Aged>) {
    print("\(celebrator.name)\(celebrator.age)岁生日快乐!")
}
//创建一个Person结构体,实现了Named和Aged协议
let birthdayPerson = Person(name: "小明", age: 21)
wishHappyBirthday(birthdayPerson) //结果:小明21岁生日快乐!

8. 可选实现要求

OC中协议定义的属性和变量有requiredoptionalSwift中你可以为协议定义optional要求,这些要求不需要被符合协议的类型实现。

  1. Optional协议要求只有在你的协议被@objc属性标记时指定。

  2. 即使你不与Objective-C交互,如果你希望指定optional要求,你仍然需要使用@objc标记你的协议。

  3. 使用@objc标记的协议只能通过类调用

根据我的理解,Swift的设计理念是没有可选的协议实现概念,但是为了保持与OC兼容性,不得已支持;所以在Swift的协议中定义可选实现的前提是该协议被@objc修饰,关于@objc

  1. @objc指示该协议暴露给OC,即可以为OC代码所用

  2. @objc修饰的协议仅仅可以被类class类型遵循

@objc protocol CounterDataSource {
    //协议可选实现的方法要求
    @optional func incrementForCount(count: Int) -> Int
    //协议可选实现的属性要求
    @optional var fixedIncrement: Int { get }
}
@objc class Counter {
    var count = 0
    var dataSource: CounterDataSource? //数据源属性,可选类型
    func increment() {
        //判断是否数据源有,数据源是否有实现可选的方法和属性
        if let amount = dataSource?.incrementForCount?(count) {
            count += amount
        } else if let amount = dataSource?.fixedIncrement? {
            count += amount
        }
    }
}
@objc class ThreeSource: CounterDataSource {
    let fixedIncrement = 3
}
var counter = Counter()
counter.dataSource = ThreeSource() //设置数据源
for i in 1...4 {
    counter.increment()
    print("\(counter.count) ") //打印:3 6 9 12
}
  1. 由于dataSource可能为nil,因此在dataSource后边加上了?标记来表明只在dataSource非空时才去调用incrementForCount方法。

  2. 即使dataSource存在,但是也无法保证其是否实现了incrementForCount方法,因此在incrementForCount方法后边也加有?标记。

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

推荐阅读更多精彩内容

  • 132.转换错误成可选值 通过转换错误成一个可选值,你可以使用 try? 来处理错误。当执行try?表达式时,如果...
    无沣阅读 1,248评论 0 3
  • 本章将会介绍 协议语法属性要求方法要求(Method Requirements)Mutating 方法要求构造器要...
    寒桥阅读 418评论 0 3
  • 1、随机数 不需要随机数种子 arc4random()%N + begin:产生begin~begin+N的随机数...
    我是小胡胡123阅读 4,159评论 0 2
  • 人生风雨,几度秋凉,化作一纸轻描淡写。 “在我住院期间,锺书只一人过日子。他每天到医院看望,常苦着脸说:‘我做坏事...
    梅琛阅读 499评论 2 2
  • 天下间,只有两个赚钱套路。1、靠能力。2、靠资源。 除非你是官二代,富二代,手里有大把的资源,你一开始赚钱,可以不...
    创业咨询阅读 296评论 0 0