性能優化8--記憶體泄露

来源:https://www.cnblogs.com/ganchuanpu/archive/2018/07/09/9282435.html
-Advertisement-
Play Games

一.根源: 記憶體泄露簡單說就是已經沒有用的資源,但是由於被其他資源引用著無法被GC銷毀。 二.記憶體泄露常見場景 1.單例導致記憶體泄露 單例的靜態特性使得它的生命周期同應用的生命周期一樣長,如果一個對象已經沒有用處了,但是單例還持有它的引用,那麼在整個應用程式的生命周期它都不能正常被回收,從而導致記憶體 ...


一.根源:

  記憶體泄露簡單說就是已經沒有用的資源,但是由於被其他資源引用著無法被GC銷毀。

二.記憶體泄露常見場景

1.單例導致記憶體泄露

   單例的靜態特性使得它的生命周期同應用的生命周期一樣長,如果一個對象已經沒有用處了,但是單例還持有它的引用,那麼在整個應用程式的生命周期它都不能正常被回收,從而導致記憶體泄露。
public class AppSettings {

    private static AppSettings sInstance;
    private Context mContext;

    private AppSettings(Context context) {
        this.mContext = context;
    }

    public static AppSettings getInstance(Context context) {
        if (sInstance == null) {
            sInstance = new AppSettings(context);
        }
        return sInstance;
    }
}
    以Activity為例,當我們啟動一個Activity,並調用getInstance(Context context)方法去獲取AppSettings的單例,傳入Activity.this作為context,這樣AppSettings類的單例sInstance就持有了Activity的引用,當我們退出Activity時,該Activity就沒有用了,但是因為sIntance作為靜態單例(在應用程式的整個生命周期中存在)會繼續持有這個Activity的引用,導致這個Activity對象無法被回收釋放,這就造成了記憶體泄露。
    為了避免這樣單例導致記憶體泄露,我們可以將context參數改為全局的上下文:
private AppSettings(Context context) {
    this.mContext = context.getApplicationContext();
}
    全局的上下文Application Context就是應用程式的上下文,和單例的生命周期一樣長,這樣就避免了記憶體泄漏。
    單例模式對應應用程式的生命周期,所以我們在構造單例的時候儘量避免使用Activity的上下文,而是使用Application的上下文
View Code

2.靜態變數導致記憶體泄露

靜態變數存儲在方法區,它的生命周期從類載入開始,到整個進程結束。一旦靜態變數初始化後,它所持有的引用只有等到進程結束才會釋放。

比如下麵這樣的情況,在Activity中為了避免重覆的創建info,將sInfo作為靜態變數:
public class MainActivity extends AppCompatActivity {

    private static Info sInfo;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        if (sInfo != null) {
            sInfo = new Info(this);
        }
    }
}

class Info {
    public Info(Activity activity) {
    }
}

Info作為Activity的靜態成員,並且持有Activity的引用,但是sInfo作為靜態變數,生命周期肯定比Activity長。所以當Activity退出後,sInfo仍然引用了Activity,Activity不能被回收,這就導致了記憶體泄露。

在Android開發中,靜態持有很多時候都有可能因為其使用的生命周期不一致而導致記憶體泄露,所以我們在新建靜態持有的變數的時候需要多考慮一下各個成員之間的引用關係,並且儘量少地使用靜態持有的變數,以避免發生記憶體泄露。當然,我們也可以在適當的時候講靜態量重置為null,使其不再持有引用,這樣也可以避免記憶體泄露。
View Code

 3.非靜態內部類導致記憶體泄露

非靜態內部類(包括匿名內部類)預設就會持有外部類的引用,當非靜態內部類對象的生命周期比外部類對象的生命周期長時,就會導致記憶體泄露。

非靜態內部類導致的記憶體泄露在Android開發中有一種典型的場景就是使用Handler,很多開發者在使用Handler是這樣寫的:

public class MainActivity extends AppCompatActivity {
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        start();
    }

    private void start() {
        Message msg = Message.obtain();
        msg.what = 1;
        mHandler.sendMessage(msg);
    }

    private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            if (msg.what == 1) {
                // 做相應邏輯
            }
        }
    };
}

熟悉Handler消息機制的都知道,mHandler會作為成員變數保存在發送的消息msg中,即msg持有mHandler的引用,而mHandler是Activity的非靜態內部類實例,即mHandler持有Activity的引用,那麼我們就可以理解為msg間接持有Activity的引用。msg被髮送後先放到消息隊列MessageQueue中,然後等待Looper的輪詢處理(MessageQueue和Looper都是與線程相關聯的,MessageQueue是Looper引用的成員變數,而Looper是保存在ThreadLocal中的)。那麼當Activity退出後,msg可能仍然存在於消息對列MessageQueue中未處理或者正在處理,那麼這樣就會導致Activity無法被回收,以致發生Activity的記憶體泄露。

通常在Android開發中如果要使用內部類,但又要規避記憶體泄露,一般都會採用靜態內部類+弱引用的方式

public class MainActivity extends AppCompatActivity {

    private Handler mHandler;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mHandler = new MyHandler(this);
        start();
    }

    private void start() {
        Message msg = Message.obtain();
        msg.what = 1;
        mHandler.sendMessage(msg);
    }

    private static class MyHandler extends Handler {

        private WeakReference<MainActivity> activityWeakReference;

        public MyHandler(MainActivity activity) {
            activityWeakReference = new WeakReference<>(activity);
        }

        @Override
        public void handleMessage(Message msg) {
            MainActivity activity = activityWeakReference.get();
            if (activity != null) {
                if (msg.what == 1) {
                    // 做相應邏輯
                }
            }
        }
    }
}

mHandler通過弱引用的方式持有Activity,當GC執行垃圾回收時,遇到Activity就會回收並釋放所占據的記憶體單元。這樣就不會發生記憶體泄露了。

上面的做法確實避免了Activity導致的記憶體泄露,發送的msg不再已經沒有持有Activity的引用了,但是msg還是有可能存在消息隊列MessageQueue中,所以更好的是在Activity銷毀時就將mHandler的回調和發送的消息給移除掉。

@Override
protected void onDestroy() {
    super.onDestroy();
    mHandler.removeCallbacksAndMessages(null);
}

非靜態內部類造成記憶體泄露還有一種情況就是使用Thread或者AsyncTask。

比如在Activity中直接new一個子線程Thread:
public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        new Thread(new Runnable() {
            @Override
            public void run() {
                // 模擬相應耗時邏輯
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
}

或者直接新建AsyncTask非同步任務:
public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        new AsyncTask<Void, Void, Void>() {
            @Override
            protected Void doInBackground(Void... params) {
                // 模擬相應耗時邏輯
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                return null;
            }
        }.execute();
    }
}

很多初學者都會像上面這樣新建線程和非同步任務,殊不知這樣的寫法非常地不友好,這種方式新建的子線程Thread和AsyncTask都是匿名內部類對象,預設就隱式的持有外部Activity的引用,導致Activity記憶體泄露。要避免記憶體泄露的話還是需要像上面Handler一樣使用靜態內部類+弱應用的方式(代碼就不列了,參考上面Hanlder的正確寫法)
View Code

4.未取消註冊或回調導致記憶體泄露

比如我們在Activity中註冊廣播,如果在Activity銷毀後不取消註冊,那麼這個剛播會一直存在系統中,同上面所說的非靜態內部類一樣持有Activity引用,導致記憶體泄露。因此註冊廣播後在Activity銷毀後一定要取消註冊。

ublic class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        this.registerReceiver(mReceiver, new IntentFilter());
    }

    private BroadcastReceiver mReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            // 接收到廣播需要做的邏輯
        }
    };

    @Override
    protected void onDestroy() {
        super.onDestroy();
        this.unregisterReceiver(mReceiver);
    }
}

在註冊觀察則模式的時候,如果不及時取消也會造成記憶體泄露。比如使用Retrofit+RxJava註冊網路請求的觀察者回調,同樣作為匿名內部類持有外部引用,所以需要記得在不用或者銷毀的時候取消註冊
View Code

5.Timer和TimerTask導致記憶體泄露

Timer和TimerTask在Android中通常會被用來做一些計時或迴圈任務,比如實現無限輪播的ViewPager
public class MainActivity extends AppCompatActivity {

    private ViewPager mViewPager;
    private PagerAdapter mAdapter;
    private Timer mTimer;
    private TimerTask mTimerTask;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        init();
        mTimer.schedule(mTimerTask, 3000, 3000);
    }

    private void init() {
        mViewPager = (ViewPager) findViewById(R.id.view_pager);
        mAdapter = new ViewPagerAdapter();
        mViewPager.setAdapter(mAdapter);

        mTimer = new Timer();
        mTimerTask = new TimerTask() {
            @Override
            public void run() {
                MainActivity.this.runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                        loopViewpager();
                    }
                });
            }
        };
    }

    private void loopViewpager() {
        if (mAdapter.getCount() > 0) {
            int curPos = mViewPager.getCurrentItem();
            curPos = (++curPos) % mAdapter.getCount();
            mViewPager.setCurrentItem(curPos);
        }
    }

    private void stopLoopViewPager() {
        if (mTimer != null) {
            mTimer.cancel();
            mTimer.purge();
            mTimer = null;
        }
        if (mTimerTask != null) {
            mTimerTask.cancel();
            mTimerTask = null;
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        stopLoopViewPager();
    }
}

當我們Activity銷毀的時,有可能Timer還在繼續等待執行TimerTask,它持有Activity的引用不能被回收,因此當我們Activity銷毀的時候要立即cancel掉Timer和TimerTask,以避免發生記憶體泄漏。
View Code

6.集合中的對象未清理造成記憶體泄露

這個比較好理解,如果一個對象放入到ArrayList、HashMap等集合中,這個集合就會持有該對象的引用。當我們不再需要這個對象時,也並沒有將它從集合中移除,這樣只要集合還在使用(而此對象已經無用了),這個對象就造成了記憶體泄露。並且如果集合被靜態引用的話,集合裡面那些沒有用的對象更會造成記憶體泄露了。所以在使用集合時要及時將不用的對象從集合remove,或者clear集合,以避免記憶體泄漏。
View Code

7.資源未關閉或釋放導致記憶體泄露

在使用IO、File流或者Sqlite、Cursor等資源時要及時關閉。這些資源在進行讀寫操作時通常都使用了緩衝,如果及時不關閉,這些緩衝對象就會一直被占用而得不到釋放,以致發生記憶體泄露。因此我們在不需要使用它們的時候就及時關閉,以便緩衝能及時得到釋放,從而避免記憶體泄露。
View Code

8.屬性動畫造成記憶體泄露

動畫同樣是一個耗時任務,比如在Activity中啟動了屬性動畫(ObjectAnimator),但是在銷毀的時候,沒有調用cancle方法,雖然我們看不到動畫了,但是這個動畫依然會不斷地播放下去,動畫引用所在的控制項,所在的控制項引用Activity,這就造成Activity無法正常釋放。因此同樣要在Activity銷毀的時候cancel掉屬性動畫,避免發生記憶體泄漏。

@Override
protected void onDestroy() {
    super.onDestroy();
    mAnimator.cancel();
}
View Code

9.WebView造成記憶體泄露

https://www.jianshu.com/p/3e8f7dbb0dc7

關於WebView的記憶體泄露,因為WebView在載入網頁後會長期占用記憶體而不能被釋放,因此我們在Activity銷毀後要調用它的destory()方法來銷毀它以釋放記憶體。

另外在查閱WebView記憶體泄露相關資料時看到這種情況:

Webview下麵的Callback持有Activity引用,造成Webview記憶體無法釋放,即使是調用了Webview.destory()等方法都無法解決問題(Android5.1之後)。

最終的解決方案是:在銷毀WebView之前需要先將WebView從父容器中移除,然後在銷毀WebView。詳細分析過程請參考這篇文章:WebView記憶體泄漏解決方法。

@Override
protected void onDestroy() {
    super.onDestroy();
    // 先從父控制項中移除WebView
    mWebViewContainer.removeView(mWebView);
    mWebView.stopLoading();
    mWebView.getSettings().setJavaScriptEnabled(false);
    mWebView.clearHistory();
    mWebView.removeAllViews();
    mWebView.destroy();
}
View Code

 

 三.優化方案

構造單例的時候儘量別用Activity的引用;
靜態引用時註意應用對象的置空或者少用靜態引用;
使用靜態內部類+軟引用代替非靜態內部類;
及時取消廣播或者觀察者註冊;
耗時任務、屬性動畫在Activity銷毀時記得cancel
文件流、Cursor等資源及時關閉;
Activity銷毀時WebView的移除和銷毀。




 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 為什麼不使用xml繪製Andoird的UI? 類型不安全 非空不安全 xml迫使你在很多佈局中寫很多相同的代碼 設備在解析xml時會耗費更多的cpu運行時間和電池 最重要的時,它允許任何代碼重用 簡單案例 findViewById() LayoutParams 使用 描述佈局參數,其中寬高都是 設置 ...
  • 添加依賴 Color 不透明的紅色 Dimensions 使用 和`sp dip(dipValue) sp(spValue)` applyRecursively() applyRecursively()應用於lambda表達式的本身,然後遞歸地運用在他的子view 以上表示,該 中的 的`textS ...
  • Alamofire框架的使用一 —— 基本用法 對於使用Objective-C的開發者,一定非常熟悉AFNetworking這個網路框架。在蘋果推出的Swift之後,AFNetworking的作者專門用Swift來編寫一個類似AFNetworking的網路框架,稱為Alamofire。Alamofi ...
  • Live Photo開發,瞭解之後可以將其應用於開發過程中。 ...
  • 1.什麼是ANR 在Android上,如果你的應用程式有一段時間響應不夠靈敏,系統會向用戶顯示一個對話框,這個對話框稱作應用程式無響應(ANR:Application Not Responding)對話框。用戶可以選擇讓程式繼續運行,但是,他們在使用你的應用程式時,並不希望每次都要處理這個對話框。因 ...
  • 1為什麼要做性能優化? 手機性能越來越好,不用糾結這些細微的性能? Android每一個應用都是運行的獨立的Dalivk虛擬機,根據不同的手機分配的可用記憶體可能只有(32M、64M等),所謂的4GB、6GB運行記憶體其實對於我們的應用不是可以任意索取 詳情:http://10.158.0.33/bbs ...
  • App效果: 功能和交互簡單描述: 針對微信使用用戶每天的零碎時間來進行天氣,新聞要點等查看,免去了打開其他App來查看; 針對每一天可以設置一項重要任務計劃,可開啟通知提醒,讓每一天任務簡化,免去太多任務計劃導致不能按時執行; 很多人在每一天玩手機時間過長,耽誤了工作生活,專註時間功能開啟,即統計 ...
  • DBFlow新版使用,該實例涉及功能:1.資料庫增刪改查(操作封裝),2.同/非同步+事物操作,3.資料庫升級(新增表+新增欄位+預設值設置等)+自定義資料庫存儲路徑... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...