IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 移动开发 -> Android之学习Handler原理的总结和理解 -> 正文阅读

[移动开发]Android之学习Handler原理的总结和理解

有一段时间查资料看这个了,下面就简单记录一下我自己的理解

一、Handler工作流程

通常,我们使用Handler的时候会先new一个Handler,然后调用send或者post来处理,类似这种:
请添加图片描述
那么handler发送消息之后实际是怎么工作呢?

之前听了一节享学课堂关于Handler原理介绍的课程,里面老师给了这么一张图,我一直以来觉得这张图很好理解。

在这里插入图片描述
先看右上角,handler发送消息后,消息队列就像有传送带一样一个一个接收并处理这些发来的消息,而让这传送带动起来的东西就是Looper。Looper就相当于一个驱动,让这个传送带动起来。

这里说一下,handler发送消息是按优先级排的,往往优先级高的消息先被发送到消息队列。而消息队列实际上是根据时间先后顺序排队的单链表,是按时间先后来处理消息。

消息入消息队列的步骤大概是:
Handler(send)发送消息—>调用messageQueue.enqueMessage方法—>messageQueue消息入消息队列

Looper驱动这个消息队列来处理消息:
Looper—>调用loop()方法来驱动—>messageQueue.next()—>dispatchMessage方法—>handleMessage

二、Handler有关常见问题

1、子线程到主线程通信都有哪些方式?子线程到主线程通信的原理?

答:handler,runonUIThread,协程,asycntask,rxjava等等

原理就是上面handler的原理

2、一个线程可以有几个Handler?

答:多个

理解:每个activity里面都可以new,这些activity都在一个线程里面。

3、一个线程有几个Looper?如何保证?

答:一个;

理解:Looper启动的时候(Looper.perpare)会new一个Looper,一旦创建,就不能再new(如果再次new,会抛出异常)。(这只能保证存进table数组的(key,value)即(ThreadLocal,Looper)中的key,value一一对应)。
在这里插入图片描述

本质原因:线程ThreadLocal里面有有一个localValue变量,一个线程只有一个value,只会用一次。

app一旦启动,Looper就会loop()

如果是子线程的handler:Looper的启动要使用loop();

4、Handler内存泄漏原因?为什么其他的内部类没有说过这个问题?

答:原因:Handler内部类持有外部类引用
因为通常使用handler是为了操作activity的view相关,避免不了持有。

理解:为什么其他内部类(像RecyclerView的ViewHolder)没有?
我觉得可能是因为static静态类不持有外部类对象,还有就是因为Handler原理的原因。

持有实际上是因为ActivityThread 持有-> Looper持有 -> MessageQueue持有 -> Message持有 -> handler持有 -> activity

所以有时还在等待未处理的(未释放的)message会泄漏,所以new Handler那里会有黄色警告。
在这里插入图片描述
在这里插入图片描述

5、使用Handler导致内存泄露的解决方法?

这里是看的这篇文章:
Android中使用Handler造成内存泄露的分析和解决

方法一:通过程序逻辑来进行保护。

1.在关闭Activity的时候停掉你的后台线程。线程停掉了,就相当于切断了Handler和外部连接的线,Activity自然会在合适的时候被回收。

2.如果你的Handler是被delay的Message持有了引用,那么使用相应的Handler的removeCallbacks()方法,把消息对象从消息队列移除就行了。

方法二:将Handler声明为静态类。

静态类不持有外部类的对象,所以你的Activity可以随意被回收。代码如下:

static class MyHandler extends Handler {
    @Override
    public void handleMessage(Message msg) {
        View view=null;
        this.click(view);
    }
}

但其实没这么简单。使用了以上代码之后,你会发现,由于Handler不再持有外部类对象的引用,导致程序不允许你在Handler中操作Activity中的对象了。
所以你需要在Handler中增加一个对Activity的弱引用(WeakReference):

static class MyHandler extends Handler {
    WeakReference<Activity > mActivityReference;

    MyHandler(Activity activity) {
        mActivityReference= new WeakReference<Activity>(activity);
    }

    @Override
    public void handleMessage(Message msg) {
        final Activity activity = mActivityReference.get();
        if (activity != null) {
            mImageView.setImageBitmap(mBitmap);
        }
    }
}

延伸:什么是弱引用WeakReference?

WeakReference弱引用,与强引用(即我们常说的引用)相对,它的特点是,GC在回收时会忽略掉弱引用,即就算有弱引用指向某对象,但只要该对象没有被强引用指向(实际上多数时候还要求没有软引用,但此处软引用的概念可以忽略),该对象就会在被GC检查到时回收掉。
对于上面的代码,用户在关闭Activity之后,就算后台线程还没结束,但由于仅有一条来自Handler的弱引用指向Activity,所以GC仍然会在检查的时候把Activity回收掉。这样,内存泄露的问题就不会出现了。

6、为何主线程可以new Handler?如果想要在子线程中new Handler要做些哪些准备?

答:主线程每个app最早执行的函数就是ActivityThread中的Main函数,而在这个函数里面就是初始化了Looper。

子线程中想要new Handler就必须初始化Looper,即先prepare,再loop()

三、持续更新中…

如有错误,欢迎指正!!!

  移动开发 最新文章
Vue3装载axios和element-ui
android adb cmd
【xcode】Xcode常用快捷键与技巧
Android开发中的线程池使用
Java 和 Android 的 Base64
Android 测试文字编码格式
微信小程序支付
安卓权限记录
知乎之自动养号
【Android Jetpack】DataStore
上一篇文章      下一篇文章      查看所有文章
加:2021-10-18 17:30:27  更:2021-10-18 17:30:41 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 22:37:31-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码