Swift | Protocol 协议特性

下面列举的都是一些比较偏僻的,特有的,很多人不知道的协议特性。
更多查看:https://segmentfault.com/a/1190000010065953

  • 虽然我们不能在协议中直接提供属性和方法的默认实现,但是我们可以通过协议扩展来达到此目的。

  • 甚至可以在扩展中添加协议里没有定义过的方法和属性。

  • 在协议中使用 static 关键字来声明类型方法,如果实现该协议的类型是 class 类型,则在实现类中除了用 static 来修饰类型方法外,也可以使用 class关键字。区别是 static 修饰的属性或方法不能被子类复写,class 修饰的可以被子类复写。

  • 如果协议中定义的实例方法会改变遵循该协议的类型的实例,那么需要在该方法前加 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()
// lightSwitch 现在的值为 .On
  • 协议中定义get、set属性
protocol MyProtocol {
    var prop: Int { get set }
    var propertyWithImplementation: String { get }
    func foo()
}

extension MyProtocol {
    var propertyWithImplementation: String {
        return "foo"
    }
    
    func foo() {
        print(prop)
    }
}

class MyClass: MyProtocol {
    var prop: Int = 29
}

  • 协议中还可以要求遵循协议的类型实现指定的构造器:
protocol SomeProtocol {
    init(someParameter: Int)
}

// 如果类为final,则不必加required标识
class SomeClass: SomeProtocol {
    required init(someParameter: Int) {
        // initializer implementation goes here
    }

  • Swift 可以采用 & 符号将多个协议进行组合。typealias Property = Named & Aged

  • 协议还可以与类进行组合:

class City: Location, Named {
    var name: String
    init(name: String, latitude: Double, longitude: Double) {
        self.name = name
// Location协议实例化
        super.init(latitude: latitude, longitude: longitude)
    }
}
  • 判断某个对象是否是某个类型用 is。在扩展协议的时候,也可以指定一些限制条件,只有遵循协议的类型满足这些限制条件时,才能获得协议扩展提供的默认实现。
extension Collection where Iterator.Element: TextRepresentable {
    var textualDescription: String {
        let itemsAsText = self.map { $0.textualDescription }
        return "[" + itemsAsText.joined(separator: ", ") + "]"
    }
}

let circulars = [Circular(radius: 1), Circular(radius: 2)]
print(circulars.textualDescription)
// [The circular's radius is 1, The circular's radius is 2]
  • Swift 是通过协议扩展提供默认实现来到达可选的目的。但需要注意的是,标记为 @objc 的 protocol 只能被 class 实现,不能被 struct 和 enum 类型实现,而且实现它的 class 中的方法也必须被标注为 @objc,或者整个类就是继承自 NSObject。
@objc protocol CounterDataSource {
    @objc optional func incrementForCount(count: Int) -> Int
    @objc optional var fixedIncrement: Int { get }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,390评论 19 139
  • 132.转换错误成可选值 通过转换错误成一个可选值,你可以使用 try? 来处理错误。当执行try?表达式时,如果...
    无沣阅读 5,090评论 0 3
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,930评论 18 399
  • 本章将会介绍 协议语法属性要求方法要求(Method Requirements)Mutating 方法要求构造器要...
    寒桥阅读 3,114评论 0 3
  • 小时候她最羡慕家庭幸福的孩子,特别不爱回家,不是邀我去她家玩,就是住在别的小朋友家。 我那时候也羡慕她,晚上吃过晚...
    盐7阅读 1,202评论 0 0