Swift - 协议底层

首先我们来看一段代码

protocol DrawProtocol {
    func draw()
}

class Student: DrawProtocol {
    var x: Int = 0
    var y: Int = 0
    func draw() {
        
    }
}

struct Point: DrawProtocol {
    var x: Int = 0
    var y: Int = 0
    func draw() {
        
    }
}

let p = Point()
let s = Student()
let draws: [DrawProtocol] = [p,s]

那么请问各位看官, draws中存储的是什么呢?
事实上,在这种情况下,变量 draws 中存储的元素是一种特殊的数据类型:Existential Container。因为: 无法确定 p , s 的内存大小!


[DrawProtocol].png

Existential Container

Existential Container是编译器生成的一种特殊的数据类型,用于管理遵守了相同协议的协议类型。因为这些数据类型的内存空间尺寸不同,使用 Extential Container 进行管理可以实现存储一致性。

结构如下:


ExistentialContainer.png

首位3个词作为 Value Buffer, 每个词包含8个字节 (存储的可能是值,也可能是指针)

Small Value(存储空间小于等于 Value Buffer),可以直接内联存储在 Value Buffer 中。
Large Value(存储空间大于 Value Buffer),当值的数量大于3个属性或者总尺寸超过valueBuffer的占位,则会在堆区分配内存进行存储,Value Buffer 只存储对应的指针, 指针指向了堆空间 (Swift 采用了 Indirect Storage With Copy-On-Write 技术进行了优化。)

Value Buffer.png

Copy-On-Write 这种技术可以提高内存指针利用率,降低堆区内存消耗,从而实现性能提升。该技术的原理是:拷贝时仅仅拷贝 Extension Container,当修改值时,先检测引用计数,如果引用计数大于 1,则开辟新的堆区内存

1 个词作为 Value Witness Table (管理协议类型的生命周期)
由于协议类型的具体类型不同,其内存布局也不同,Value Witness Table 则是对协议类型的生命周期进行专项管理,从而处理具体类型的初始化、拷贝、销毁。

Value Witness Table.png

1 个词作为 Protocol Witness Table (管理协议类型的方法调用)
在 Class 中,基于继承关系的多态是通过 Virtual Table 实现的;在 POP 中,没有继承关系,因为无法使用 Virtual Table 实现基于协议的多态,取而代之的是 Protocol Witness Table。每个结构体会创造Protocol Witness Table表中,内部包含指针,指向方法!


Protocol Witness Table.png

内存分布如下:

1. payload_data_0 = 0x0000000000000004,
2. payload_data_1 = 0x0000000000000000,
3. payload_data_2 = 0x0000000000000000,
4. instance_type = 0x000000010d6dc408 ExistentialContainers`type    
       metadata for ExistentialContainers.Car,
5. protocol_witness_0 = 0x000000010d6dc1c0 
       ExistentialContainers protocol witness table for 
       ExistentialContainers.Car:ExistentialContainers.Drivable 
       in ExistentialContainers

在Swift编译器中,通过Existential Container实现的伪代码如下:

func drawACopy(local :Drawable) {
 local.draw()
}
let val :Drawable = Point()
drawACopy(val)

//existential container的伪代码结构
struct ExistContDrawable {
 var valueBuffer:(Int, Int, Int)
 var vwt:ValueWitnessTable
 var pwt:DrawableProtocolWitnessTable
}

// drawACopy方法生成的伪代码
func drawACopy(val:ExistContDrawable) { //将existential container传入
 var local = ExistContDrawable()  //初始化container
 let vwt = val.vwt //获取value witness table,用于管理生命周期
 let pwt = val.pwt //获取protocol witness table,用于进行方法分派
 local.type = type 
 local.pwt = pwt
 vwt.allocateBufferAndCopyValue(&local, val)  //vwt进行生命周期管理,初始化或者拷贝
 pwt.draw(vwt.projectBuffer(&local)) //pwt查找方法,这里说一下projectBuffer,因为不同类型在内存中是不同的(small value内联在栈内,large value初始化在堆内,栈持有指针),所以方法的确定也是和类型相关的,我们知道,查找方法时是通过当前对象的地址,通过一定的位移去查找方法地址。
 vwt.destructAndDeallocateBuffer(temp) //vwt进行生命周期管理,销毁内存
}

Protocol Type 存储属性

在Swift中class的实例和属性都存储在堆区,Struct实例在栈区! 如果包含指针属性则存储在堆区,Protocol Type如何存储属性?
小的数据则通过Existential Container内联实现。那么存在堆区的数据,又是如何处理Copy呢?

protocol Drawable { func draw() }
class Point {
    var x1: CGFloat = 0
    var x2: CGFloat = 0
    var y1: CGFloat = 0
    var y2: CGFloat = 0
}

struct Student: Drawable {
 var p: Point
 func draw() { }
}

let s1 = Student(p: Point())
let s2 = s1
copy.png

将新的Exsitential Container的valueBuffer指向同一个value即创建指针引用,但是如果要改变值怎么办?我们知道Struct值的修改和Class不同,Copy是不应该影响原实例的值的!

这里用到了一个技术叫做Indirect Storage With Copy-On-Write,即优先使用内存指针。通过提高内存指针的使用,来降低堆区内存的初始化。降低内存消耗。在需要修改值的时候,会先检测引用计数检测,如果有大于1的引用计数,则开辟新内存,创建新的实例。在对内容进行变更的时候,会开启一块新的内存。
伪代码如下:

struct Line :Drawable {
 var storage : Point
 init() { 
  storage = Point()
 }
 func draw() { }
 mutating func move() {
 // 如过存在多份引用,则开启新内存,否则直接修改
   if !isUniquelyReferencedNonObjc(&storage) {
     storage = Point(storage) //柯里化
   }
  }
}

这样实现的目的:通过多份指针去引用同一份地址的成本远远低于开辟多份堆内存。

静态多态 Static Polymorphism

protocol Drawable {
 func draw()
}

struct Line: Drawable {
    var x = 0
    func draw() {}
}

struct Point: Drawable {
    var y = 0
    func draw() {}
}
func drawACopy(local :Drawable) {
 local.draw()
}

let line = Line()
drawACopy(line)

let point = Point()
drawACopy(point)

关于 Virtual Table 和 Protocol Witness Table 的区别,个人理解:
它们都是一个记录函数地址的列表(即函数表),只是它们的生成方式是不同的。
对于 Virtual Table,在编译时,子类的函数表是通过对父类函数表进行拷贝、覆写、插入等操作生成的。
对于 Protocol Witness Table,在编译时,函数表是通过识别当前类型对协议的实现,直接生成的。

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

推荐阅读更多精彩内容