Android 内存泄漏实战:从排查到修复的完整指南

通过实战示例和工具使用,帮助开发者理解、排查和修复 Android 应用中的内存泄漏问题
1. 什么是内存泄漏?

定义:内存泄漏是指程序中已动态分配的内存由于某种原因未能释放,导致系统内存的浪费,最终可能导致应用崩溃或性能下降。

在 Android 中的表现:Activity 或 Fragment 被销毁后,仍然被其他对象持有引用,导致无法被垃圾回收器回收

2. 内存泄漏的常见场景

以下是 Android 开发中常见的内存泄漏场景:

(1)静态变量持有 Context 引用

复制代码
public class LeakExample {
    private static Context sContext;

    public static void setContext(Context context) {
        sContext = context; // 静态变量持有 Activity 引用,导致内存泄漏
    }
}

(2)非静态内部类持有外部类引用

复制代码
public class MainActivity extends AppCompatActivity {
    private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            // 处理消息
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 发送延迟消息
        mHandler.sendEmptyMessageDelayed(0, 10000);
    }
}

(3)未取消的注册监听器

复制代码
public class MainActivity extends AppCompatActivity {
    private SensorManager sensorManager;
    private Sensor sensor;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
        sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
        sensorManager.registerListener(this, sensor, SensorManager.SENSOR_DELAY_NORMAL);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 未取消注册监听器,导致内存泄漏
        // sensorManager.unregisterListener(this);
    }
}

(4)匿名内部类持有外部类引用

复制代码
public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        new Thread(new Runnable() {
            @Override
            public void run() {
                // 长时间运行的任务
                while (true) {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }).start();
    }
}

3. 内存泄漏的排查工具
(1)Android Profiler

使用 Memory Profiler 监控内存使用情况。

捕获堆转储(Heap Dump)并分析内存中的对象。

(2)LeakCanary

集成 LeakCanary 库,自动检测内存泄漏。

添加依赖:

复制代码
dependencies {
    debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.9.1'
}

初始化:

复制代码
public class MyApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        if (LeakCanary.isInAnalyzerProcess(this)) {
            return;
        }
        LeakCanary.install(this);
    }
}

4. 内存泄漏的修复方法

(1)避免静态变量持有 Context 引用

使用 WeakReferenceApplication Context

复制代码
public class LeakExample {
    private static WeakReference<Context> sContextRef;

    public static void setContext(Context context) {
        sContextRef = new WeakReference<>(context);
    }
}

(2)使用静态内部类 + WeakReference

将非静态内部类改为静态内部类,并使用 WeakReference 持有外部类引用。

复制代码
public class MainActivity extends AppCompatActivity {
    private static class MyHandler extends Handler {
        private WeakReference<MainActivity> mActivityRef;

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

        @Override
        public void handleMessage(Message msg) {
            MainActivity activity = mActivityRef.get();
            if (activity != null) {
                // 处理消息
            }
        }
    }

    private MyHandler mHandler = new MyHandler(this);
}

(3)及时取消注册监听器

onDestroy() 中取消注册监听器。

复制代码
@Override
protected void onDestroy() {
    super.onDestroy();
    sensorManager.unregisterListener(this);
}

(4)避免匿名内部类持有外部类引用

将匿名内部类改为静态内部类,或使用 WeakReference

复制代码
public class MainActivity extends AppCompatActivity {
    private static class MyRunnable implements Runnable {
        private WeakReference<MainActivity> mActivityRef;

        MyRunnable(MainActivity activity) {
            mActivityRef = new WeakReference<>(activity);
        }

        @Override
        public void run() {
            MainActivity activity = mActivityRef.get();
            if (activity != null) {
                // 执行任务
            }
        }
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        new Thread(new MyRunnable(this)).start();
    }
}

5. 实战示例:修复一个内存泄漏问题

场景描述

一个 Activity 中使用了 Handler 发送延迟消息,导致 Activity 无法被回收。

修复步骤

将 Handler 改为静态内部类。

使用 WeakReference 持有 Activity 引用。

在 onDestroy() 中移除所有消息。

修复代码

复制代码
public class MainActivity extends AppCompatActivity {
    private MyHandler mHandler = new MyHandler(this);

    private static class MyHandler extends Handler {
        private WeakReference<MainActivity> mActivityRef;

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

        @Override
        public void handleMessage(Message msg) {
            MainActivity activity = mActivityRef.get();
            if (activity != null) {
                // 处理消息
            }
        }
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mHandler.sendEmptyMessageDelayed(0, 10000);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        mHandler.removeCallbacksAndMessages(null); // 移除所有消息
    }
}

6. 总结

内存泄漏是 Android 开发中的常见问题,可能导致应用崩溃或性能下降。

通过工具(如 Android Profiler 和 LeakCanary)可以快速定位内存泄漏。

使用 WeakReference、静态内部类、及时取消注册等方法可以有效修复内存泄漏。

通过本文的实战指南,你可以掌握 Android 内存泄漏的排查和修复技巧,提升应用的质量和性能。

相关推荐
teacher伟大光荣且正确4 小时前
Qt Creator 配置 Android 编译环境
android·开发语言·qt
飞猿_SIR7 小时前
Android Exoplayer 实现多个音视频文件混合播放以及音轨切换
android·音视频
HumoChen997 小时前
GZip+Base64压缩字符串在ios上解压报错问题解决(安卓、PC模拟器正常)
android·小程序·uniapp·base64·gzip
沙振宇11 小时前
【HarmonyOS】ArkTS开发应用的横竖屏切换
android·华为·harmonyos
橙子1991101613 小时前
Kotlin 中的作用域函数
android·开发语言·kotlin
zimoyin13 小时前
Kotlin 懒初始化值
android·开发语言·kotlin
枣伊吕波14 小时前
第六节第二部分:抽象类的应用-模板方法设计模式
android·java·设计模式
萧然CS14 小时前
使用ADB命令操作Android的apk/aab包
android·adb
_extraordinary_18 小时前
MySQL 事务(二)
android·数据库·mysql
鸿蒙布道师1 天前
鸿蒙NEXT开发动画案例5
android·ios·华为·harmonyos·鸿蒙系统·arkui·huawei