Swift - 協(xié)議底層

首先我們來看一段代碼

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 中存儲的元素是一種特殊的數(shù)據(jù)類型:Existential Container。因為: 無法確定 p , s 的內(nèi)存大小!


[DrawProtocol].png

Existential Container

Existential Container是編譯器生成的一種特殊的數(shù)據(jù)類型嫉拐,用于管理遵守了相同協(xié)議的協(xié)議類型哩都。因為這些數(shù)據(jù)類型的內(nèi)存空間尺寸不同,使用 Extential Container 進行管理可以實現(xiàn)存儲一致性婉徘。

結(jié)構(gòu)如下:


ExistentialContainer.png

首位3個詞作為 Value Buffer漠嵌, 每個詞包含8個字節(jié) (存儲的可能是值,也可能是指針)

Small Value(存儲空間小于等于 Value Buffer)盖呼,可以直接內(nèi)聯(lián)存儲在 Value Buffer 中儒鹿。
Large Value(存儲空間大于 Value Buffer),當(dāng)值的數(shù)量大于3個屬性或者總尺寸超過valueBuffer的占位几晤,則會在堆區(qū)分配內(nèi)存進行存儲约炎,Value Buffer 只存儲對應(yīng)的指針, 指針指向了堆空間 (Swift 采用了 Indirect Storage With Copy-On-Write 技術(shù)進行了優(yōu)化。)

Value Buffer.png

Copy-On-Write 這種技術(shù)可以提高內(nèi)存指針利用率蟹瘾,降低堆區(qū)內(nèi)存消耗圾浅,從而實現(xiàn)性能提升。該技術(shù)的原理是:拷貝時僅僅拷貝 Extension Container憾朴,當(dāng)修改值時狸捕,先檢測引用計數(shù),如果引用計數(shù)大于 1伊脓,則開辟新的堆區(qū)內(nèi)存

1 個詞作為 Value Witness Table (管理協(xié)議類型的生命周期)
由于協(xié)議類型的具體類型不同府寒,其內(nèi)存布局也不同,Value Witness Table 則是對協(xié)議類型的生命周期進行專項管理报腔,從而處理具體類型的初始化株搔、拷貝、銷毀纯蛾。

Value Witness Table.png

1 個詞作為 Protocol Witness Table (管理協(xié)議類型的方法調(diào)用)
在 Class 中纤房,基于繼承關(guān)系的多態(tài)是通過 Virtual Table 實現(xiàn)的;在 POP 中翻诉,沒有繼承關(guān)系炮姨,因為無法使用 Virtual Table 實現(xiàn)基于協(xié)議的多態(tài),取而代之的是 Protocol Witness Table碰煌。每個結(jié)構(gòu)體會創(chuàng)造Protocol Witness Table表中舒岸,內(nèi)部包含指針,指向方法!


Protocol Witness Table.png

內(nèi)存分布如下:

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實現(xiàn)的偽代碼如下:

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

//existential container的偽代碼結(jié)構(gòu)
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,因為不同類型在內(nèi)存中是不同的(small value內(nèi)聯(lián)在棧內(nèi)夜焦,large value初始化在堆內(nèi)壳澳,棧持有指針),所以方法的確定也是和類型相關(guān)的茫经,我們知道巷波,查找方法時是通過當(dāng)前對象的地址,通過一定的位移去查找方法地址科平。
 vwt.destructAndDeallocateBuffer(temp) //vwt進行生命周期管理褥紫,銷毀內(nèi)存
}

Protocol Type 存儲屬性

在Swift中class的實例和屬性都存儲在堆區(qū),Struct實例在棧區(qū)! 如果包含指針屬性則存儲在堆區(qū)瞪慧,Protocol Type如何存儲屬性髓考?
小的數(shù)據(jù)則通過Existential Container內(nèi)聯(lián)實現(xiàn)。那么存在堆區(qū)的數(shù)據(jù)弃酌,又是如何處理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即創(chuàng)建指針引用,但是如果要改變值怎么辦?我們知道Struct值的修改和Class不同妓湘,Copy是不應(yīng)該影響原實例的值的!

這里用到了一個技術(shù)叫做Indirect Storage With Copy-On-Write查蓉,即優(yōu)先使用內(nèi)存指針。通過提高內(nèi)存指針的使用榜贴,來降低堆區(qū)內(nèi)存的初始化豌研。降低內(nèi)存消耗妹田。在需要修改值的時候,會先檢測引用計數(shù)檢測鹃共,如果有大于1的引用計數(shù)鬼佣,則開辟新內(nèi)存,創(chuàng)建新的實例霜浴。在對內(nèi)容進行變更的時候晶衷,會開啟一塊新的內(nèi)存。
偽代碼如下:

struct Line :Drawable {
 var storage : Point
 init() { 
  storage = Point()
 }
 func draw() { }
 mutating func move() {
 // 如過存在多份引用阴孟,則開啟新內(nèi)存晌纫,否則直接修改
   if !isUniquelyReferencedNonObjc(&storage) {
     storage = Point(storage) //柯里化
   }
  }
}

這樣實現(xiàn)的目的:通過多份指針去引用同一份地址的成本遠遠低于開辟多份堆內(nèi)存。

靜態(tài)多態(tài) 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)

關(guān)于 Virtual Table 和 Protocol Witness Table 的區(qū)別永丝,個人理解:
它們都是一個記錄函數(shù)地址的列表(即函數(shù)表)锹漱,只是它們的生成方式是不同的。
對于 Virtual Table慕嚷,在編譯時凌蔬,子類的函數(shù)表是通過對父類函數(shù)表進行拷貝、覆寫闯冷、插入等操作生成的砂心。
對于 Protocol Witness Table,在編譯時蛇耀,函數(shù)表是通過識別當(dāng)前類型對協(xié)議的實現(xiàn)辩诞,直接生成的。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末纺涤,一起剝皮案震驚了整個濱河市译暂,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌撩炊,老刑警劉巖外永,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異拧咳,居然都是意外死亡伯顶,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進店門骆膝,熙熙樓的掌柜王于貴愁眉苦臉地迎上來祭衩,“玉大人,你說我怎么就攤上這事阅签∑海” “怎么了?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵政钟,是天一觀的道長路克。 經(jīng)常有香客問我樟结,道長,這世上最難降的妖魔是什么精算? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任狭吼,我火速辦了婚禮,結(jié)果婚禮上殖妇,老公的妹妹穿的比我還像新娘。我一直安慰自己破花,他們只是感情好谦趣,可當(dāng)我...
    茶點故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著座每,像睡著了一般前鹅。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上峭梳,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天舰绘,我揣著相機與錄音,去河邊找鬼葱椭。 笑死捂寿,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的孵运。 我是一名探鬼主播秦陋,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼治笨!你這毒婦竟也來了驳概?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤旷赖,失蹤者是張志新(化名)和其女友劉穎顺又,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體等孵,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡稚照,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了俯萌。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片锐锣。...
    茶點故事閱讀 40,090評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖绳瘟,靈堂內(nèi)的尸體忽然破棺而出雕憔,到底是詐尸還是另有隱情,我是刑警寧澤糖声,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布斤彼,位于F島的核電站分瘦,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏琉苇。R本人自食惡果不足惜嘲玫,卻給世界環(huán)境...
    茶點故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望并扇。 院中可真熱鬧去团,春花似錦、人聲如沸穷蛹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽肴熏。三九已至鬼雀,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間蛙吏,已是汗流浹背源哩。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留鸦做,地道東北人励烦。 一個月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像泼诱,于是被迫代替她去往敵國和親崩侠。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,033評論 2 355

推薦閱讀更多精彩內(nèi)容