String,相信你們都不陌生,咱們在編寫程序時,使用String類型還算比較多。那麼你常用它,是否真的「瞭解」它?請帶着問題,一步一步揭開它神祕的面紗,看看它究竟何許「人」也!git
1、思考github
在 Swift 開發使用字符串的過程當中,你是否有思考過如下問題?編程
字符串 str一、str2 的底層存儲有什麼不一樣?sass
若是對 str一、str2 進行拼接操做,str一、str2 的底層存儲又會發生什麼變化?微信
若是你能準確地回答以上問題,那說明對 Swift 字符串的底層存儲機制仍是比較瞭解的。數據結構
2、1 個字符串變量佔用多少內存?ide
方法 1:MemoryLayout工具
首先,能夠藉助 Swift 自帶的 MemoryLayout 來測試一下學習
方法 2:彙編測試
另外,咱們也能夠藉助一個強有力的底層分析助手—彙編語言,來窺探一下 String 的底層存儲
實際上分析其餘語法、系統庫的底層,均可以藉助彙編語言
另外,不只僅是 Swift,C、C++、OC 的底層分析,依然能夠藉助彙編語言
跟微軟的 Visual Studio 同樣,Xcode 也內置了很是方便的反彙編功能,能夠輕鬆查看每一句代碼對應的彙編指令,打開反彙編界面的步驟以下
在某一行須要調試的代碼打上斷點(反彙編界面會在斷點調試狀態下顯示出來)
菜單:Debug
> Debug Workflow
> Always Show Disassembly
Assembly
譯爲彙編, Disassembly
譯爲反彙編運行程序,看到反彙編界面
若是你的反彙編經驗十足,根據第 1六、17 行的彙編就能夠推敲出來,String 是佔用 16 個字節
彙編的內容太多了,由於時間和篇幅關係,文章裏並不會對每一句彙編指令進行詳細地講解,更多的是想說明彙編的重要性。
3、字符串的底層存儲
窺探內存
此前我寫了個能夠窺探 Swift 變量內存的小工具:https://github.com/CoderMJLee/Mems
如今用它來窺探下字符串的 16 字節裏面,究竟存儲着什麼數據
Mems.memStr(ofVal:)
默認狀況下按照 8 個字節一組來顯示內存數據
alignment: .one
是按照 1 個字節一組來顯示內存數據字符 '0'~'9' 的 ASCII 值是 0x30~0x39,認真觀察最初 str1 的 16 個字節數據,你發現了什麼?
它直接將全部字符的 ASCII 值存儲在 str1 的 16 字節中
最後 1 個字節 0xea 中的 0xa 就是字符的數量,也是共 10 個字符
拼接
能夠發現,當對 str1 進行拼接 "ABCDE" 的時候
它最終是將 "0123456789ABCDE"十五個字符的 ASCII 值都存儲在了 str1 的 16 字節中
最後 1 個字節 0xef 中的 0xf 就是字符的數量,也是共 15 個字符
能夠看得出來,目前 16 個字節已經存滿了,那若是再拼接 1 個字符呢?
能夠看到,str1 裏面存儲的數據發生了很是大的變化,每個字符的 ASCII 值不見了,
那裏面的 16 字節具體是什麼含義呢?
若是一開始初始化的時候(未拼接以前),字符串的內容就是超過 15 個字符呢?
相信你能猜到是這個結果
第27行的str1
仍是有所區別
若是對 str2 進行拼接操做
不難發現:這時 str2 的 16 字節又發生了變化,跟 第27行的str1
是有點類似的
如何解決上述疑問?
上述的種種疑問,光看打印出來的內存數據是沒法解決的,可是均可以利用【!!!彙編!!!】來解決,分析彙編指令,立馬就得出結論,由於文章的篇幅有限,平時工做也比較忙,我把上述問題的詳細剖析過程錄製成了長達 2 個多小時的視頻,有興趣的朋友能夠用 1.5~2 倍速度觀看
連接:https://pan.baidu.com/s/1AkS3K1ZKP8zyxhlhLRaBkA
視頻對於沒有彙編基礎的朋友來講,可能會有點難度,最好挑一個頭腦清醒的時間去觀看
4、最後
彙編語言雖然是編程中的基礎語言,但確是咱們用到最多的計算機語言,應用領域也不只僅是在你的工做當中,平常生活中也是能夠用到的,好比你還能玩轉軟件破解、開外掛等。我做爲一個在IT行業摸爬滾打了數年的人,也確實積累了比較多的經驗及資源,平時也會給你們作分享,若是想獲取更多免費的編程學習資源及乾貨,能夠手動添加微信:19950277730!來跟我一塊兒探索編程的世界吧!