對待棘手bug,新手與大牛的差距在哪里?

一行代碼引發周邊童鞋的Xcode內存爆炸。作為一名喜歡探究到底的工程師,豈能袖手旁觀?來自高德的濤瀾童鞋,給出了一個樣本式的解決思路。下面就讓我們一起走進“案發現場”。

問題描述:

自上上周起,團隊中陸續有iOS開發抱怨電腦特別卡。有細心的同學發現,因為Xcode占用了約6-7G內存,而部分mac只有8G內存,所以內存爆滿引起卡頓。

而部分同學的mac是16G內存的,比如我(嘲諷臉),因為內存充足沒感覺到卡。

但這個問題影響團隊的開發效率,所以需要去解決問題。

內存對比:

在沐浴更衣焚香、殺進程、清緩存后,分別拉取相鄰的812版本代碼和816版本代碼分別編譯,得到結論:

812調試時,占用2G內存

816調試時,占用6.8G內存

吐槽:

對于這個數據,我們內心是拒絕接受的。有如下2點吐槽:

如果代碼亂申請內存,那么內存爆掉的應該是模擬器或真機。而不該是Xcode

如果當前版本新增10w行代碼(其實不到),對總代碼量增長不超10%,Xcode內存怎么可能翻兩翻。

所以我們覺得,這一定是蘋果的鍋,我們不背

但是不管是誰的鍋,肯定是代碼或者配置觸發的,分析還要繼續。

分析方法選擇:

擺在我們面前有2個分析方法:

找代碼:通過二分法,編譯不同日期的版本,找到引發問題的那次提交,確定是哪個改動引起

找內存:分析增大的內存是什么,根據增大的內容分析問題出在哪。

如果使用方法1,編譯一次代碼需要15分鐘,假設問題是某一行代碼引起的,估計需要找一天。如果是某多行代碼組合影響的問題,時間會更長。而且就算找到代碼,也未必知道原理是什么。

所以我選擇**方法2**,不行再退到方法1

分析步驟:

我在run的時候發現:

812初次打開代碼內存1G以內,編譯運行時內存2G,關閉Xcode后再打開內存2G

816初次打開代碼內存1G以內,編譯運行時內存6G,關閉Xcode后再打開內存6G

關閉Xcode后再打開,此時Xcode并沒有run,所以推測他在做一件事:讀緩存

緩存文件:

大家都知道,Xcode編譯一個新工程會很慢,但是第二次編譯就很快。那是因為他把編譯結果存到了緩存文件中。第二次編譯只讀文件不編譯自然就快了。

緩存文件存儲在“/Users/你的用戶名/Library/Developer/Xcode/DerivedData”目錄下

812和816版本的緩存文件對比如下:

初步可以看出,緩存文件數量一致,但是大小差距很大。所以下一步就是來找茬:到底誰變大了

經過一番尋找,發現每個類會生成三個文件:

.o文件:二進制對象文件,不多說

.d文件:文本文件,記錄該類依賴的所有文件路徑

.dia文件:未知二進制文件,但是變大的就是它

.dia是有一部分變大了,一部分沒變。嘗試用二進制工具打開讀了一下,有驚喜:

這不就是warning嘛

我的吐槽又來了:

是誰!站出來!寫了4個G的warning!

繼續分析:

那具體是什么導致的warning呢,面對幾千個.dia文件,我內心是崩潰的。

幸好找基友溝通,剛好他做了代碼warning掃描,發現816比812只是某組代碼多了107個warning,其他組沒變化,而且是nonnull相關warning,并不重要所以沒追究。

我們找到107處warning的代碼,查看提交記錄,就是在大家反饋卡頓之前。貌似就是它了。我們把warning解了,clean重新編譯,問題得解。

問題雖解,但是遺留2個問題:

怎么就提交了107個warning?

區區107個warning。為啥會導致內存飆升?我們還剩幾百個warning為啥沒問題?

問題1:

引發107個warning的只有一行代碼

對于nonnull相關warning蘋果的潛規則是這樣的:

自Xcode6起提供的新功能,可以申明一個函數的參數是必傳的(nonnull)還是可選的(nullable) ,這會讓代碼更嚴謹,我們是推薦使用的

兼容老代碼:整個頭文件都沒有nonnull/nullable申明的,編譯沒毛病

對新代碼高要求:只要給代碼中添加了一個nonnull/nullable,剩余的代碼也必須添加,否則其他每個接口就會有warning

所以,這次涉案的代碼是個舊工具類,有107個函數。新增的一行代碼添加了nonnull。于是產生了107個warning

問題2:

舉個例子,有A B C三個類

A.h有一個warning,其.dia文件中會如下信息:

insert '_Nullable' if the pointer may be null

insert '_Nonnull' if the pointer should never be null

A.m文件絕對路徑

A.h文件絕對路徑

A.m文件第幾行引用了A.h,存在warning

warning在A.h的位置

warning描述是:pointer is missing a nullability type specifier (_Nonnull, _Nullable, or _Null_unspecified)

fix的兩種方法:

總之,一處warning的信息大約是1k

如果B引用了A,則B的.dia文件包含如上所有信息,以及多個B的文件路徑,即B的描述信息超過A

如果C引用了B,而B在頭文件中引用了A,則C的描述信息超過B

所以

在工程上,107warning的文件,dia約130k。

所有直接間接引用的文件數量大概2500,單個文件都超過130k。文件大小約350M。

加上模擬器有2個cpu架構(i386/x86_64),會生成2份文件,緩存中還有個聚合的dgph文件。以及文件在內存中結構化后占用的內存空間。

所以最終翻了幾倍,達到4G的內存占用是可以理解的。

結論:

不要忽略warning,特別是頭文件中的warning,會被多處引用導致過大的描述信息

頭文件中盡量不要import頭文件,會造成過度的引用,放大問題。

后續:

818版本已經fix了core中的所有nonnull問題。后續逐步將warning清零

fix后內存占用如圖

PS:這是蘋果的bug么?我覺得還是自己挖坑把自己埋了。

遇到棘手的bug,你的解決思路是什么呢?歡迎在評論區留言,一起交流學習。

原文鏈接

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,002評論 6 542
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,400評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,136評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,714評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,452評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,818評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,812評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,997評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,552評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,292評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,510評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,035評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,721評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,121評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,429評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,235評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,480評論 2 379

推薦閱讀更多精彩內容