一.瞅瞅String是個(gè)啥
stirng是個(gè)結(jié)構(gòu)體
@frozen public struct String
二.做1個(gè)測(cè)試
var str1: String = "0123456789"
print(MemoryLayout.stride(ofValue: str1)) //16
let ptr1 = withUnsafeMutablePointer(to: &str1) {$0} //0x000000016da55ad0
看一下ptr1的內(nèi)容
再看一下ASCII表
圖片.png
發(fā)現(xiàn)0x000000016da55ad0里面直接存儲(chǔ)了0123456789這幾個(gè)字符。
8-16位0xea00000000003938
多出來(lái)了一個(gè)0xea
,我們看下那是什么
- e 代表類(lèi)型標(biāo)識(shí)崭庸, 即該字符串是直接存儲(chǔ)了內(nèi)容信息
- a等于十進(jìn)制的10奶陈, 代表這個(gè)字符串占用了10個(gè)字節(jié)徘键。
總結(jié):
這種簡(jiǎn)單字符串嘁灯,最多能存儲(chǔ)15個(gè)字節(jié)內(nèi)容悍抑,剩下一個(gè)字節(jié)用來(lái)描述字符串的類(lèi)型肯适。和OC的tagPointer技術(shù)非常一樣群井。
三.再做1個(gè)測(cè)試
var str2: String = "0123456789ABCEDF"
print(MemoryLayout.stride(ofValue: str2)) //16
let ptr2 = withUnsafeMutablePointer(to: &str2) {$0} //0x000000016ba39ad0
看看超過(guò)15個(gè)字節(jié)的字符串, 內(nèi)部已經(jīng)不直接存儲(chǔ)字符串的具體內(nèi)容了
看看匯編吧
如果小于16,就跳轉(zhuǎn)到其他函數(shù)
x0 = 0x0000000104071370 "0123456789ABCEDF" = 字符串的真實(shí)地址
x1 = 0x0000000000000010 = 十進(jìn)制的16 = 字符串的長(zhǎng)度
x2 = 0x0000000000000001
字符串的真正位置: 代碼段
圖片.png
總結(jié):大于16位的字符串汰瘫,一般占用16個(gè)字節(jié)。
- 0xd000000000010 d是個(gè)一個(gè)標(biāo)志位擂煞,10代表字符串的長(zhǎng)度
- 0x80000001043d9350: 減去#0x7fffffffffffffe0 = 0x1043D9370, 即字符串真正的地址
四.append的情況
注意:只要是swfit的字符串混弥,其內(nèi)容都處于TEXT_cstring中。
注意:str1和str2是個(gè)指針对省,在棧區(qū)
- 當(dāng)小于16位的字符串蝗拿,使用類(lèi)似oc的tagPointer技術(shù),將字符串從test段讀取到棧指針里面。如果append其他字符蒿涎,只要不超過(guò)16哀托,就繼續(xù)添加。
- 當(dāng)大于16位的字符串劳秋,如果append則會(huì)在堆空間創(chuàng)建新的字符串仓手,然后棧指針的第8-16位指向新的空間。