Handler/Handler.post引发内存泄漏

本文详细探讨了Android中Handler机制的使用误区,特别是handler.removeCallbacks(null)与handler.removeCallbacksAndMessages(null)的区别。前者仅能移除未运行的Runnable,而后者能彻底清除所有回调和消息,有效防止内存泄漏。文章提供了具体的代码示例和解决方案,包括使用静态内部类、弱引用和静态Runnable,以及在Activity的onDestroy方法中正确移除消息。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  GC ROOT static HandlerCenter.mHandlerList
-- Android removeCallbacksAndMessages(null)不起作用; handler.removeCallbacks(null)不起作用;
  调用handler.removeCallbacksAndMessages(null);可以清空当前handler的所有回调和所有消息。想当然的认为handler.removeCallbacks(null)可以清空所有回调,实则不然。
  handler.removeCallbacks(null)不会移除任何回调。因为已经处于运行的runnable包含在message中已经移交给handler进行处理了,而Handler.removeCallbacks(Runnable r)只能移除未运行的runnable,所以导致无法停止runnbale的运行,最后失效。

handler.removeCallbacksAndMessages(null):
Remove any pending posts of callbacks and sent messages whose obj is token. If token is null, all callbacks and messages will be removed.

@Override
    protected void onDestroy() {
        if (handler != null && handler.getLooper() == Looper.getMainLooper()){
              handler.removeCallbacksAndMessages(null);
           }
        super.onDestroy();
    }
if (myHandler != null) {
            myHandler.removeCallbacksAndMessages(null);
        }

- 解决办法
 1.最直接的思路就是避免使用非静态内部类。使用Handler的时候,放在一个新建的文件中来继承Handler或者使用静态的内部类来替代。静态内部类不会隐含的持有外部类的引用,因此这个activity也就不会出现内存泄漏问题。
 2.如果你需要在Handler内部调用外部Activity的方法,你可以让这个Handler持有这个Activity的弱引用,这样便不会出现内存泄漏的问题了。
 3.另外,对于匿名类Runnable,我们同样可以设置成静态的,因为静态内部类不会持有外部类的引用。
 4.注意:如果使用Handler发送循环消息,最好是在Activity的OnDestroy方法中调用mLeakHandler.removeCallbacksAndMessages(null);移除消息。(这不是解决内存泄漏的方法)
 5.两种解决办法如下:
 a.弱引用(WeakReference)
  public class SampleActivity extends Activity {

  /**
   * Instances of static inner classes do not hold an implicit
   * reference to their outer class.
   * 弱引用的方式
   */
    private static class MyHandler extends Handler {
    private final WeakReference<SampleActivity> mActivity;
    public MyHandler(SampleActivity activity) {
      mActivity = new WeakReference<SampleActivity>(activity);
    }

    @Override
    public void handleMessage(Message msg) {
      SampleActivity activity = mActivity.get();
      if (activity != null) {
         //to Something
      }
    }
 }

 b.静态
 //定义成static的,因为静态内部类不会持有外部类的引用
  private final MyHandler mHandler = new MyHandler(this);
  private static final Runnable sRunnable = new Runnable() {
      @Override
      public void run() {//to something}
  };

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    mHandler.postDelayed(sRunnable, 1000 * 60 * 10);
    finish();
  }
}
  removeCallbacks只能停止处于即将运行状态的messages,这里messages里面包含就是Runnables,如果Runnbale已经启动start(),则removeCallbacks是无法停止的。

### Handler.post 的使用方法 `Handler.post()` 是 `Handler` 提供的一个重要功能,用于将任务提交到与其关联的消息队列中执行。此方法允许开发者在指定线程的上下文中运行一段代码片段。 #### 方法签名 以下是 `post` 方法的主要形式: ```java public final boolean post(Runnable r) ``` 该方法接受一个实了 `Runnable` 接口的对象作为参数,并将其放入消息队列中等待执行[^5]。 当调用 `handler.post(r)` 时,`r.run()` 将会在与 `Handler` 关联的线程上被执行。如果未显式绑定其他 Looper,则默认会是在主线程上执行。 #### 示例代码 下面是一个简单的示例展示如何利用 `Handler.post()` 来更新 UI: ```java // 创建一个基于主线程Looper的Handler实例 Handler handler = new Handler(Looper.getMainLooper()); // 定义一个匿名内部类实 Runnable接口的任务 Runnable task = new Runnable() { @Override public void run() { // 更新UI组件的操作放在这里,比如修改TextView的内容 TextView textView = findViewById(R.id.textView); textView.setText("Updated by Handler!"); } }; // 把这个任务加入到Handler管理的消息队列里去异步执行 if (!handler.hasMessages(0)) { handler.post(task); } ``` 上述代码展示了怎样通过 `Handler.post()` 调度一个更改界面控件状态的动作至主线程环境下来完成安全地刷新视图的工作流程[^6]。 另外还存在延时版本的方法可以设定延迟时间后再执行相应指令: ```java long delayMillis = 2000; // 延迟两秒 handler.postDelayed(new Runnable(){ @Override public void run(){} },delayMillis ); ``` 以上就是关于 Android 中 `Handler.post` 的基本介绍及其典型应用场景之一——跨线程间传递数据并最终影响到图形用户界面上的表方式[^7]。 ### 注意事项 尽管 `Handler.post()` 非常方便易用,但在某些特殊场景下需要注意避免内存泄漏等问题的发生。例如长时间持有对外部对象(如 Activity 或 Fragment)强引用可能导致这些本应被回收销毁掉的东西无法释放资源从而造成潜在风险。因此建议尽可能采用弱引用模式或者及时清除回调函数等方式加以防范[^8]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值