美文网首页
iOS函数派发机制

iOS函数派发机制

作者: 你猜我猜不猜你猜我猜不猜 | 来源:发表于2022-09-05 11:35 被阅读0次

函数派发是编程语言中管理函数调用的过程,关于函数派发机制大概可以分成以下几种

  • 静态派发 Static Dispatch
  • 函数表派发 V-Table Dispatch
  • 消息派发 Message Dispatch

静态派发 Static Dispatch

静态派发显然是速度最快的,不单单是因为需要调用的指令集会更少,而且编译器还能够有很大的优化空间。

值类型的函数是静态派发,而引用类型的函数需要static和final来标识为静态派发。

一个函数如果不能被重写,它将被静态派发,因为该类型的函数不能被重写,所以这个函数只有一个函数实现,在需要使用时,只要直接跳转到这个存储函数实体的内存地址执行就行了。

函数表派发 V-Table Dispatch

引用类型默认使用这种派发方式,之所以使用这种方式是因为类需要支持继承。函数表通过生成对重写方法和非重写方法的正确调用来帮助继承类。

函数表派发就是通过函数表来查找相应的函数地址。每个类在创建时都会创建一个函数表,用来记录函数的指针。同时子类在创建时也会创建一个函数表,如果函数是override的,则使用一个新的指针,用于区分父类中相同函数的指针。如果这个函数是父类中有且没被override的,则存储的就是原先的指针。

class ParentClass {
    func method1() {}
    func method2() {}
}
class ChildClass: ParentClass {
    override func method2() {}
    func method3() {}
}
image.png
let obj = ChildClass()
obj.method2()

函数调用过程:

  • 读取对象 0xB00 的函数表
  • 读取函数指针的索引. 在这里, method2 的索引是1(偏移量), 也就是 0xB00 + 1
  • 跳到 0x222 (函数指针指向 0x222)

消息派发 Message Dispatch

Runtime

OC是一种动态语言,主要用的就是消息派发机制,OC拥有的Runtime可以做很多操作,比如

  • 使用isMemberOfClass检查实例对象是否属于特定类型的类。或者如果想检查实例对象是否属于其继承层次结构中的特定类,可以使用isKindOfClass.
  • 检查类是否可以调用某个函数 respondsToSelector
  • 在运行时通过swizzling修改函数的实现,也可以通过isa-swizzling修改对象。
  • 使用在运行时添加方法实现class_addMethod

OC调用函数

Person *p = Person.new;
[p sayHello];

其实本质上调用了:

((void (*)(id,SEL))objc_msgSend)(p,@selector(sayHello));

简单说一下这个函数干了啥:

  1. 先判断实例对象是否为空
  2. 去函数缓存表里找这个函数,找到执行
  3. 缓存表里没有就去类本身的函数列表里找,找到执行并将函数加入函数缓存表
  4. 类本身也没找到,去父类的函数列表里找,找到执行并将函数加入函数缓存表,未找到再去父类的父类里找,直到根类NSObject
  5. 如果都没找到,进行三次转发,如果消息被处理结束流程,没被处理App crash

现在我们知道了OC依托Runtime是一种非常灵活的动态语言,并且可以在运行时更改方法实现、添加方法实现等。

dynamic和@objc

Swift本身是没有Runtime的,但是Swift可以利dynamic和@objc关键字用嫁接到OC去使用Runtime。

  • dynamic 将函数开启动态性(例如,在方法调配或 KVO 相关代码中使用)
  • @objc 将该函数公开给OC。

在 Swift 中,需要用作selector对象的方法必须用声明@objc,因为target-action的机制仍然是用OC编写的,所以这些函数需要暴露给OC使用。

顺便一提iOS的各个热修方案也是依靠这种动态性,如JSPatch。

JSPatch

JSPatch是早期用于iOS平台上的轻量级热修框架,主要利用OC动态语言的特性,将js文件内的js代码以字符串的形式传递给OC,OC 通过 Runtime 接口调用和替换 OC 函数,这是最基础的原理。如下图示:

image.png

最后

最后总结一下
1.如果不需要多态,静态派发。
2.如果需要覆写,函数表派发。
3.如果需要对Objective-C可见和动态性,消息派发。

最后的最后简单举个例子

protocol Noisy {
     func makeNoise() -> Int  //函数表派发
}

extension Noisy {
    func makeNoise() -> Int { return 0 }  //函数表派发
    func isAnnoying() -> Bool { return true}  //直接派发
}

class Animal: Noisy {
    func makeNoise() -> Int { return 1 } //函数表派发
    func isAnnoying() -> Bool { return false } //函数表派发
    @objc func sleep() {} //消息派发
}

extension Animal {
    func eat() {} //静态派发 extension内的函数不能被继承
    @objc func getWild() {} //消息派发
}

struct rectangle {
    func getArea() { } //静态派发
}

相关文章

  • Swift 函数派发机制

    编程语言 函数派发机制有三种: 【原文链接】 直接派发 函数表派发 消息机制派发 函数派发是程序判断使用哪种途径去...

  • iOS函数派发机制

    函数派发是编程语言中管理函数调用的过程,关于函数派发机制大概可以分成以下几种 静态派发 Static Dispat...

  • iOS知识复习笔记(16)---swift相关

    一、函数的派发方式 swift函数的派发机制有三类:static直接派发(静态)派发、table函数派发,mess...

  • 浅谈Swift派发机制

    函数的派发机制分为:静态派发(直接派发)、函数表派发、消息派发 1、Swift中所有ValueType(值类型:S...

  • Swift中的函数派发机制

    函数派发机制指的是程序如何找到函数并执行操作的机制。各种各样不同的需求导致不同的函数派发机制。有时可能希望函数直接...

  • Swift - 三种函数派发

    Swift 的函数派发可以分为 静态 和 动态 两种机制,而动态派发又分为 函数表派发 和 消息派发 。 Swif...

  • Swift - 函数的派发方式.

    编译型语言有三种基础的派发方式: 静态派发;函数表派发;消息机制派发(动态派发).我们都知道Objective-C...

  • 初探Swift函数的派发方式

    派发方式 (Types of Dispatch ) 函数派发就是程序判断使用哪种途径去调用一个函数的机制. 每次函...

  • Swift 函数派发机制

    原文:Method Dispatch in Swift作者:Brian King 派发机制是程序判断如何去调用函数...

  • Swift 函数派发机制

    函数派发方式 能够在编译期确定执行方法的方式叫做静态分派 Static dispatch,无法在编译期确定,只能在...

网友评论

      本文标题:iOS函数派发机制

      本文链接:https://www.haomeiwen.com/subject/phfmdrtx.html