手把手教你在Android Studio 3.0上分析內存泄漏

封面

戳我下載 Android Studio 3.0

這個不用梯子我會告訴你嗎

1.寫在前面

Google在上周發布了Android Studio 3.0的正式版本,周四早晨在上班的地鐵上就看到群里在沸沸揚揚的討論關于3.0版本的各種坑,啊,不對,各種特性,到公司之后就迫不及待的更新了3.0版本,嗯,還算順利,只遇到了一個坑,一切都在happy的進行著。

什么,你以為我想要寫遇到的坑是什么,呵呵噠,我才不會告訴你,等等。。。手里的板磚先放下,一會說還不行嗎,今天我們主要來聊聊如何在Android Studio 3.0上分析內存泄漏,文章的內容很簡單,但是自己摸索還是需要一些時間的,所以就在這里記錄下來分享給大家。

2.強大的Android Profiler

戳這里查看官方文檔

在3.0版本中,android使用了新的性能分析工具Android Profiler來代替原有的Android Monitor,使用方式和原來類似,都可以分析CPU、內存和網絡的使用情況,但是功能強大了很多。

開始使用

還記得我之前寫過一篇文章《Android 使用RxLifecycle解決RxJava內存泄漏》,本文將以這篇文章里的Demo為例,使用Android Studio 3.0再次分析一下內存泄漏。

首先點擊工具欄中的Profile按鈕將待分析的App安裝到設備上,也可以直接安裝,在AS底部選擇Android Profiler按鈕:

將待分析的APP安裝到設備上

可以看到有下面的提示,大概意思是不能在當前進程進行更高級的分析:

不能在當前進程進行更高級的分析

點擊Run Configuration進去看看,發現不能勾選開關,提示gradle插件版本太低,需要2.4以上版本才可以,嗯,那就更新一下:

更新gradle插件版本

已經更新到3.0版本了,可以勾選開關了,點擊確定:

dependencies {
    classpath 'com.android.tools.build:gradle:3.0.0'
}
勾選開關

又來一個警告,大概意思是說,你的gradle版本已經升級到3.0了,需要和26.0.2版本的構建工具搭配才更好,好好好,聽你的:

更新26.0.2版本的構建工具

更新完成之后,需要再次運行一下App,如果還提示不能進行更高級的分析,請重啟Android Studio,重啟還不好,沒關系,反正今天也用不到它,不要打我,下面來看下正常的Android Profiler:

Android Profiler

點擊MEMORY進入內存詳情,在這里可以實時查看內存的占用情況:

內存詳情

內存泄漏分析

我們先寫個會發生內存泄漏的程序分析一下:

public class RxLifecycleComponentsActivity extends RxAppCompatActivity {

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_rxlifecycle);
        ButterKnife.bind(this);

        initData();
    }

    private void initData() {
        // 每隔1s執行一次事件
        Observable.interval(1, TimeUnit.SECONDS)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Observer<Long>() {
                    @Override
                    public void onSubscribe(@NonNull Disposable d) {

                    }

                    @Override
                    public void onNext(@NonNull Long aLong) {
                        Log.i("接收數據", String.valueOf(aLong));
                    }

                    @Override
                    public void onError(@NonNull Throwable e) {

                    }

                    @Override
                    public void onComplete() {

                    }
                });
    }
}

很簡單,每隔1s發送一條數據,因為關閉Activity之后沒有取消訂閱,RxJava還繼續持有Activity的引用,所以在內存回收的時候,該Activity不會被回收,由此引發內存泄漏。

下面反復打開關閉頁面5次,然后手動GC(點擊左上角的垃圾桶圖標),發現內存占用并沒有減少:

內存泄漏分析

分析一下當前的內存堆棧情況(點擊垃圾桶圖標右側的圖標):

分析內存堆棧情況

選擇按包名查找,找到當前測試的Activity,發現存在5個實例,由此可見,內存已經發生了泄漏:

內存泄漏

防止內存泄漏

修改一下上面的代碼,在關閉Activity時取消訂閱:

public class RxLifecycleComponentsActivity extends RxAppCompatActivity {

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_rxlifecycle);
        ButterKnife.bind(this);

        initData();
    }

    private void initData() {
        // 每隔1s執行一次事件
        Observable.interval(1, TimeUnit.SECONDS)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .compose(this.<Long>bindUntilEvent(ActivityEvent.DESTROY))
                .subscribe(new Observer<Long>() {
                    @Override
                    public void onSubscribe(@NonNull Disposable d) {

                    }

                    @Override
                    public void onNext(@NonNull Long aLong) {
                        Log.i("接收數據", String.valueOf(aLong));
                    }

                    @Override
                    public void onError(@NonNull Throwable e) {

                    }

                    @Override
                    public void onComplete() {

                    }
                });
    }
}

反復打開頁面5次,手動GC,看下當前的堆棧情況,可以看到當前已經沒有RxLifecycleComponentsActivity的實例存在了:

無內存泄漏

OK,到這里,在Android Studio 3.0上分析內存泄漏就學習完了,趕快去動手試試吧!

3.更新Android Studio遇到的問題

編譯的時候報錯:

Error:(41, 0) Cannot set the value of read-only property 'outputFile' for ApkVariantOutputImpl_Decorated{apkData=Main{type=MAIN, fullName=debug, filters=[]}} of type com.android.build.gradle.internal.api.ApkVariantOutputImpl.

發現是在gradle里打包輸出apk的代碼出的問題,原代碼是這樣的:

applicationVariants.all { variant ->
    variant.outputs.each { output ->
        def file = output.outputFile
        String apkName = "APK_NAME" + defaultConfig.versionName.replace(".", "_") + ".apk"
        output.outputFile = new File(file.parent, apkName)
    }
}

修改成這樣就可以了:

applicationVariants.all { variant ->
    variant.outputs.all {
        outputFileName = "APK_NAME" + defaultConfig.versionName.replace(".", "_") + ".apk"
    }
}

4.寫在最后

戳我下載 Android Studio 3.0

戳我下載本文使用的測試Demo

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

推薦閱讀更多精彩內容