19.Protocol与静态派发、动态派发

一、 场景一

我们定义这样一个Protocol

protocol DescriptionProtocol {
    var name: String { get }
}

添加拓展方法 printDescription

extension DescriptionProtocol {
    func printDescription() {
        print("我是:\(name)")
    }
}

新建类 Apple 遵循协议 DescriptionProtocol

class Apple: DescriptionProtocol {
    var name: String {
        return "苹果"
    }

    func printDescription() {
        print("\(name)好好吃啊!")
    }
}

新建类 People 遵循协议 DescriptionProtocol

分别创建实例对象

1.1 思考

1.2 再通过这样的方法来调用:

是不是感觉很奇怪?

  • aApple.printDescription()

    • 这种方式直接调用了类定义的实例方法,而不是协议中实现的方法,是动态派发的

  • showDescription

    • 中调用的是Protocol中实现的函数,是静态派发

  • printDescription在协议中和类中虽然同名,但却是毫不相干的函数,存储的区域也各不相同

二、 场景二

我们将函数在协议中声明一下下,又会有什么结果呢?

结果:

  • aApple.printDescription()

    • 和场景一是一样的,是动态派发

  • showDescription

    • 调用了协议中声明的函数,是动态派发

三、 总结

由继承而得到的多态表现,本质上是把所有的方法的实现都放在一张里,每次调用都必须先在表中查询,可以称之为动态调用。 因此才有了覆写的概念:方法被覆写,即当子类调用该方法时,会使用子类的实现而非父类的。

而协议与类有所不同。协议所定义的方法,也和类一样,是动态调用的;但那些未出现在定义中,而仅出现于 Extension 中的方法是静态调用的,即这个方法属于协议,不属于协议的遵从者。

性能上也是静态优于动态。像是一些高频调用的函数我们可以通过静态的方式来做提高效率。动态调用虽然步骤会多一些,比如OC的消息转发机制,但是充满灵活性,在业务开发中会经常用到。

参考

static-vs-dynamic-dispatch-in-swift

Last updated

Was this helpful?