Android原生開發(fā)--用Valgrind排查內(nèi)存問題

身為一名iOS開發(fā),最近在幫安卓的同事研究排查安卓原生C++代碼內(nèi)存問題的方案... 在iOS中非常簡(jiǎn)單书幕,但到了安卓中就有點(diǎn)復(fù)雜了新荤。折騰了好幾天,發(fā)現(xiàn)網(wǎng)上很多人發(fā)的內(nèi)容不是很全面按咒,反正我跑不起來迟隅。后來在stackoverflow上看到一個(gè)帖子,驗(yàn)證了一下是可行的方案励七,但使用起來有點(diǎn)繁瑣智袭。正好想學(xué)shell腳本,就邊學(xué)邊做掠抬,做了個(gè)小工具 ValgrindHelperForAndroid, 我把它提交到了GitHub上吼野。

現(xiàn)在有很多安卓應(yīng)用因?yàn)樾阅芑虬踩矫嬖颍瑫?huì)通過JNI調(diào)用原生代碼两波。排查原生的C瞳步、C++代碼時(shí)可以使用Valgrind。Valgrind工具包括Memcheck(用于檢測(cè)C和C++中與內(nèi)存相關(guān)的錯(cuò)誤)腰奋、Cachegrind(緩存分析器)单起、Massif(堆分析器)和其他幾種工具。Valgrind在Linux開發(fā)中應(yīng)用廣泛劣坊,但在安卓開發(fā)中用起來比較麻煩嘀倒,官方文檔和網(wǎng)上的資料也比較少。這就是這個(gè)工具誕生的原因局冰。
關(guān)于Valgrind這個(gè)工具测蘑,這里就不多做介紹了,大家可以參考官網(wǎng)或google.

原創(chuàng)文章康二,如需轉(zhuǎn)載請(qǐng)?jiān)谙旅媪粞宰屛抑??碳胳。不留言不在開頭標(biāo)明出處鏈接的壞同學(xué),1字1元索賠??


  • 開始之前的說明:
    1.你的安卓設(shè)備需要root.(有興趣完善這個(gè)腳本的朋友可以嘗試一下把這條限制去掉)
    2.目前沫勿,在下面的電腦和安卓設(shè)備中測(cè)試成功了(歡迎提交Pull request或在留言里發(fā)出你運(yùn)行成功的環(huán)境和設(shè)備挨约,我會(huì)添加到這里),其他PC和設(shè)備可能需要修改腳本藕帜。如果需要修改腳本烫罩,可以參考原理中的內(nèi)容。
    • macOS 10.13 華為榮耀3C(H30-T00) 4.4.2

*原理和FAQ在Wiki里洽故。
如果你優(yōu)化了這個(gè)腳本的兼容性或修改了bug贝攒,(或者有能力把這篇文章和Wiki里的內(nèi)容翻譯成英文??)歡迎提交Pull request??

使用

  • 將Valgrind安裝到安卓設(shè)備上:

    • 1.下載ValgrindHelperForAndroid.zip和對(duì)應(yīng)你手機(jī)cpu架構(gòu)的Valgrind壓縮包:
      ARMv7,
      ARM64.
    • 2.解壓ValgrindHelperForAndroid.zip,把Valgrind壓縮包放到ValgrindHelperForAndroid文件夾中时甚。就像這樣:
      [圖片上傳失敗...(image-20520f-1512456953110)]
    • 3.在終端中執(zhí)行下面的命令:
    cd <ValgrindHelperForAndroid Path>  
    ./sss_valgrind_android.sh -I
    
    • 4.出現(xiàn)下面的提示說明安裝成功:
      DONE, install valgrind success!
  • 用Valgrind啟動(dòng)安卓app:

    • 1.安裝app隘弊。建議用ValgrindHelperForAndroid中的valgrind_test工程測(cè)試,make project, debug app荒适。
    • 2.在終端中執(zhí)行下面的命令:
    cd <ValgrindHelperForAndroid Path>
    ./sss_valgrind_android.sh
    
    • 3.輸入Valgrind命令參數(shù)梨熙。(可以直接輸入回車,用Valgrind默認(rèn)的參數(shù))
    • 4.輸入app的包名刀诬。(valgrind_test工程為com.sunshushu.test)
    • 5.輸入app的MainActivity的名稱咽扇。(valgrind_test工程為MainActivity)
    • 6.輸入工程中帶符號(hào)表的庫(kù)的路徑。(valgrind_test工程為valgrind_test/app/build/intermediates/cmake/debug/obj/armeabi-v7a)
    • 7.等待app啟動(dòng)。這個(gè)過程可能會(huì)很長(zhǎng)质欲,設(shè)備中途可能會(huì)無響應(yīng)或黑屏树埠,你可以先去喝杯咖啡??(先不要關(guān)閉終端
    • 8.測(cè)試app。valgrind_test中嘶伟,點(diǎn)擊app的“MEMORY ISSUES”按鈕怎憋,會(huì)產(chǎn)生3個(gè)內(nèi)存問題。然后點(diǎn)擊“EXIT”按鈕九昧,正常退出app.
      (由于Valgrind在應(yīng)用結(jié)束之前要進(jìn)行一些工作绊袋,如果直接強(qiáng)制關(guān)閉app可能導(dǎo)致內(nèi)存問題排查不全面,所以這里設(shè)計(jì)了EXIT按鈕铸鹰。)
  • 強(qiáng)制關(guān)閉app癌别,并從安卓設(shè)備取回Valgrind的日志。
    在終端中繼續(xù)執(zhí)行下面的命令:
    DONE
    等待終端中出現(xiàn)下面的提示時(shí)蹋笼,到ValgrindHelperForAndroid文件夾中查看Valgrind日志即可规个。
    DONE. Check the log(s) in ...
    如果你在上面的第8步中點(diǎn)擊“EXIT”按鈕關(guān)閉了app,這里的強(qiáng)制關(guān)閉不會(huì)影響Valgrind徹底排查內(nèi)存問題姓建。

  • 查看日志诞仓。在日志中搜索app的包名,看一下是不是有內(nèi)存問題吧速兔。
    (日志文件名中的數(shù)字是線程ID(PID)墅拭。)

原理

  • 安裝Valgrind:
    解壓valgrind壓縮包,將壓縮包中的Inst/Data/Local/Inst拷貝到設(shè)備上的/data/local涣狗。
    (你也可以自己去Valgrind官網(wǎng)下載源碼谍婉,自己編譯。)
  • 用Valgrind啟動(dòng)安卓app:
    1.引導(dǎo)用戶輸入后面要用到的信息镀钓。
    2.把下面的內(nèi)容寫到文件中穗熬,(我把它命名為了start_valgrind.sh),然后移到設(shè)備的/data/local文件夾中丁溅。
#!/system/bin/sh
PACKAGE="${PACKAGE_NAME}"
VGPARAMS='${VALGRIND_PARAMS}'
export TMPDIR=/data/data/$PACKAGE
exec /data/local/Inst/bin/valgrind $VGPARAMS $*

3.替換app的庫(kù)唤蔗。測(cè)試的時(shí)候我發(fā)現(xiàn),即使是debug的安裝包窟赏,Android Studio安裝到設(shè)備上的app庫(kù)也不包含符號(hào)表妓柜。這會(huì)導(dǎo)致導(dǎo)出的Valgrind日志中沒法顯示出問題的文件名、函數(shù)名等信息涯穷。所以這個(gè)腳本會(huì)把帶符號(hào)表的庫(kù)替換到設(shè)備的/data/data/$PACKAGE_NAME/lib中棍掐。
(如果想判斷一個(gè)庫(kù)中是否帶有符號(hào)表,可以用nm命令)
4.設(shè)置安卓設(shè)備的系統(tǒng)屬性拷况。在終端中執(zhí)行下面的命令:

adb shell setprop wrap.$PACKAGE_NAME "logwrapper /data/local/start_valgrind.sh"

5.強(qiáng)制關(guān)閉并啟動(dòng)app

adb shell am force-stop $PACKAGE_NAME
adb shell am start -a android.intent.action.MAIN -n "$PACKAGE_NAME/.$APP_MAIN_ACTIVITY"

6.用上面的命令作煌,強(qiáng)制關(guān)閉app掘殴。之后從設(shè)備上取回Valgrind日志。

adb pull "/sdcard/$LOG_PATH"

FAQ

Q: valgrind_test測(cè)試正常粟誓,但是我自己的工程中沒有產(chǎn)生日志杯巨?
A: app需要有讀寫sd卡文件的權(quán)限。

Q: 用完這個(gè)腳本之后努酸,為什么我的app啟動(dòng)速度變得特別慢?
A: 可能是因?yàn)槭褂眠@個(gè)腳本啟動(dòng)了你的app之后杜恰,直接退出了腳本获诈,沒有用這個(gè)腳本關(guān)閉app⌒暮郑可以嘗試重啟你的安卓設(shè)備舔涎。

Q: 用腳本啟動(dòng)了我的app之后,可以斷開設(shè)備和電腦的連接嗎逗爹?
A: 可以亡嫌。只要在測(cè)試完app之后,重新把設(shè)備和電腦連上掘而,繼續(xù)執(zhí)行腳本就可以了挟冠。

Q: 我不想用你提供的Valgrind,我可以自己從官網(wǎng)下載嗎袍睡?
A: 沒有任何問題知染,其實(shí)我提供的壓縮包就是官方代碼編譯的。如果你想自己編譯Valgrind斑胜,可以參考官網(wǎng)控淡,只要將編譯好的Inst文件夾放到設(shè)備/data/local中就可以了止潘。

Q: 你的腳本對(duì)輸入?yún)?shù)做的檢查太嚴(yán)格了掺炭,比如說我沒有帶符號(hào)表的庫(kù)凭戴,怎么用這個(gè)腳本呢?
A: 你可以直接在腳本后面加4個(gè)參數(shù)么夫,分別為Valgrind參數(shù)、app包名魏割、MainActivity名、帶符號(hào)表的庫(kù)文件夾路徑钞它。哪個(gè)參數(shù)不想填,可以直接留空枫攀。比如:

./sss_valgrind_android.sh "" "com.sunshushu.test" "MainActivity" ""

One more thing

感謝stackoverflow上bitek的回答

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末总放,一起剝皮案震驚了整個(gè)濱河市俯画,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌趾盐,老刑警劉巖小腊,帶你破解...
    沈念sama閱讀 216,470評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異秩冈,居然都是意外死亡本缠,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門入问,熙熙樓的掌柜王于貴愁眉苦臉地迎上來丹锹,“玉大人,你說我怎么就攤上這事芬失¢故颍” “怎么了?”我有些...
    開封第一講書人閱讀 162,577評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵棱烂,是天一觀的道長(zhǎng)锡凝。 經(jīng)常有香客問我,道長(zhǎng)垢啼,這世上最難降的妖魔是什么窜锯? 我笑而不...
    開封第一講書人閱讀 58,176評(píng)論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮芭析,結(jié)果婚禮上锚扎,老公的妹妹穿的比我還像新娘。我一直安慰自己馁启,他們只是感情好驾孔,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,189評(píng)論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著惯疙,像睡著了一般翠勉。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上霉颠,一...
    開封第一講書人閱讀 51,155評(píng)論 1 299
  • 那天对碌,我揣著相機(jī)與錄音,去河邊找鬼蒿偎。 笑死朽们,一個(gè)胖子當(dāng)著我的面吹牛怀读,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播骑脱,決...
    沈念sama閱讀 40,041評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼菜枷,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了叁丧?” 一聲冷哼從身側(cè)響起啤誊,我...
    開封第一講書人閱讀 38,903評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎拥娄,沒想到半個(gè)月后蚊锹,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,319評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡条舔,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,539評(píng)論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了乏矾。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片孟抗。...
    茶點(diǎn)故事閱讀 39,703評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡钻心,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出捷沸,到底是詐尸還是另有隱情,我是刑警寧澤痒给,帶...
    沈念sama閱讀 35,417評(píng)論 5 343
  • 正文 年R本政府宣布苍柏,位于F島的核電站,受9級(jí)特大地震影響试吁,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜烛恤,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,013評(píng)論 3 325
  • 文/蒙蒙 一缚柏、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧船惨,春花似錦、人聲如沸粱锐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,664評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽恶座。三九已至,卻和暖如春跨琳,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背脉让。 一陣腳步聲響...
    開封第一講書人閱讀 32,818評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工溅潜, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人滚澜。 一個(gè)月前我還...
    沈念sama閱讀 47,711評(píng)論 2 368
  • 正文 我出身青樓设捐,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親萝招。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,601評(píng)論 2 353

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

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,072評(píng)論 25 707
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)母赵,斷路器,智...
    卡卡羅2017閱讀 134,651評(píng)論 18 139
  • 1.FFmepg編譯環(huán)境及結(jié)構(gòu) 下載FFmepg FFmpeg配置選項(xiàng)介紹 下載gas-preprocessor....
    Jackey_song閱讀 2,229評(píng)論 2 2
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,803評(píng)論 6 342
  • 四木先生閱讀 204評(píng)論 0 0