【面试题】Handler/Runnable造成的内存泄漏

1 内存泄漏根本原因

内存泄漏的根本原因是:长生命周期的对象持有短生命周期的对象,短生命周期的对象就无法及时释放。

2 Handler错误用法

    private Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            Toast.makeText(MainActivity.this, "测试", Toast.LENGTH_SHORT).show();
        }
    };

Handler允许我们发送延时消息,如果在延时消息未处理完,而此时Handler所在的Activity被关闭,但因为上述Handler用法则可能会导致内存泄漏。首先,Handler是以内部类的形式被创建,那么它将隐性持有外部类的对象,而且,在Toast中还以MainActivity.this方式显性持有了MainActivity的对象,那么,在延时消息未处理完时,Handler无法释放外部类MainActivity的对象,从而导致内存泄漏产生。

3 Handler正确用法

package com.yds.jianshu.mobile;

import android.app.Activity;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.TextView;
import android.widget.Toast;

import java.lang.ref.WeakReference;

public class MainActivity extends AppCompatActivity{
    private static final String TAG = "[MainActivity]";
    private MyHandler handler;
    private TextView tv;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initData();

    }
    private void initData(){
        tv = findViewById(R.id.text);
        handler = new MyHandler(this);
        
    }


    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (handler!=null){
            handler.removeCallbacksAndMessages(null);
        }
    }


    private static class MyHandler extends Handler{
        private WeakReferencereference;
        public MyHandler(MainActivity activity){
            reference = new WeakReference<>(activity);
        }
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            MainActivity activity = reference.get();
            if (activity!=null){
                Toast.makeText(activity, "测试", Toast.LENGTH_SHORT).show();
                activity.tv.setText("测试");
            }
           
        }
    }
}


4 Runnable错误用法

    private Runnable mRunnable = new Runnable() {
        @Override
        public void run() {

        }
    };

一样的道理,该使用方式是创建了一个内部类,内部类隐性持有外部类对象的引用,如果Activity结束,Runnable里面的任务没有处理完,则不会释放Activity的引用,则Activity无法被回收,造成内存泄漏。

5 Runnable正确用法

    private static class MyRunnable implements Runnable{
        WeakReference reference;
        public MyRunnable(MainActivity activity){
            reference = new WeakReference<>(activity);
        }
        @Override
        public void run() {
            MainActivity activity = reference.get();
        }
    }

6 总结

  • 如果直接new一个Handler,则Handler是一个非静态内部类,它隐性地持有外部类的对象。如果外部类需要结束,但消息队列中还有消息未处理完,则Handler不会释放外部类的对象,从而造成内存泄漏。
  • 如果直接new一个Runnable,Runnable也是一个非静态内部类,它隐性地持有外部类的对象。如果外部类需要结束,但Runnable中还有任务未处理完,则Runnable不会释放外部类的对象,从而造成内存泄漏。

你可能感兴趣的:(【面试题】Handler/Runnable造成的内存泄漏)