Android之Handler、Message、MessageQueue、Looper详解2

简介: Android之Handler、Message、MessageQueue、Looper详解

Android之Handler、Message、MessageQueue、Looper详解1:https://developer.aliyun.com/article/1473571

Handler.post 和 Handler.sendMessage的区别

handler.post 和 handler.sendMessage本质上是没有区别的,都是发送一个消息到消息队列中,而且消息队列和 handler 都是依赖于同一个线程的

  • post和sendMessage本质上是没有区别的,只是实际用法中有一点差别
  • post也没有独特的作用,post本质上还是用sendMessage实现的,post只是一种更方便的用法而已

Handler.post 和 View.post的区别

相同点

在与UI线程的通信上,Handler与View,其实最终都做了同样的事情。就是将消息传递在UI线程的消息队列里,执行一些处理操作。

不同View.post方法想在非UI线程有效工作。如该方法的注释所说,必须保证该View已经被添加至窗口。

This method can be invoked from outside of the UI thread only when this View is attached to a window.另外给一个stackoverflow的例子:


Handler.post,它的执行时间基本是等同于onCreate里那行代码触达的时间;

View.post,则不同,它说白了执行时间一定是在Act#onResume发生后才开始算的;或者换句话说它的效果相当于你上面的View.post方法是写在Act#onResume里面的(但只执行一次,因为onCreate不像onResume会被多次触发);

当然,虽然这里说的是post方法,但对应的postDelayed方法区别也是类似的

new Handler()和new Handler(Looper.getMainLooper())的区别

  • 如果你不带参数的实例化:Handler handler = new Handler(); 这个会默认用当前线程的looper,一般情况是当前线程的异步线程与当前线程进行消息处理。

getMainLooper()是获取UI主线程looper,在UI线程中处理消息;如果你的Handler是要用来刷新UI的,那么就需要在主线程下运行。

一般而言new Handler(Looper.getMainLooper())用于更新UI

Handler handler = new Handler()用于当前线程与异步线程的消息处理

更新UI界面,handler要用到主线程的looper。那么在主线程 Handler handler = new Handler();如果在其他线程,也要满足这个功能的话,要Handler handler = new Handler(Looper.getMainLooper());

只进行处理消息操作。 当前线程如果是主线程的话,使用Handler handler = new Handler();

不是主线程的话,Looper.prepare(); Handler handler = new Handler();Looper.loop();或者Handler handler = new Handler(Looper.getMainLooper());

若是实例化的时候用Looper.getMainLooper()就表示放到主UI线程去处理;如果不是的话,因为只有UI线程默认Loop.prepare();Loop.loop();其他线程需要手动调用这两个,否则会报错。

Message

Message.obtain() 和Handler.obtainMessage()的区别

Message.obtain() 和Handler.obtainMessage()的区别

获取Message对象的最好方法是调用Message.obtain()或者Handler.obtainMessage(),这样是从一个可回收对象池中获取Message对象。

从整个Messge池中返回一个新的Message实例,在许多情况下使用它,因为它能避免分配新的对象,如果是这样的话,那么通过调用obtainMessage方法获取Message对象就能避免创建对象,从而减少内存的开销了

这两种方式都比直接new一个Message对象在性能上更优越

message.what、message.arg1、message.arg2、message.obj它们之间有什么区别

message.what、message.arg1、message.arg2、message.obj 他们之间有什么区别呢


what就是一般用来区别消息的,比如你传进去的时候msg.what = 3;

然后处理的时候判断msg.what == 3是不是成立的,是的话,表示这个消息是干嘛干嘛的(自己能区别开)

arg1、arg2,其实也就是两个传递数据用的,两个int值,看你自己想要用它做什么了。如果你的数据只是简单的int值,那么用这两个,比较方便。

setData(Bundle),上面两个arg是传递简单int的,这个是传递复杂数据的

msg.obj呢,这个就是传递数据了,msg中能够携带对象,在handleMessage的时候,可以把这个数据取出来做处理了。不过呢,如果是同一个进程,最好用上面的setData就行了,这个一般是Messenger类来用来跨进程传递可序列化的对象的,这个比起上面的来,更消耗性能一些。

MessageQueue

MessageQueue的数据结构是什么

数据结构是单向链表,“先进先出”

MessageQueue

MessageQueue的数据结构是什么

数据结构是单向链表,“先进先出”在项目启动之初来预测将来项目会碰到什么需求,是极其困难的。消息队列在处理过程中间插入了一个隐含的、基于数据的接口层,两边的处理过程都要实现这一接口。这允许你独立的扩展或修改两边的处理过程,只要确保它们遵守同样的接口约束

冗余

有些情况下,处理数据的过程会失败。除非数据被持久化,否则将造成丢失。消息队列把数据进行持久化直到它们已经被完全处理,通过这一方式规避了数据丢失风险。在被许多消息队列所采用的”插入-获取-删除”范式中,在把一个消息从队列中删除之前,需要你的处理过程明确的指出该消息已经被处理完毕,确保你的数据被安全的保存直到你使用完毕。

扩展性

因为消息队列解耦了你的处理过程,所以增大消息入队和处理的频率是很容易的;只要另外增加处理过程即可。不需要改变代码、不需要调节参数。扩展就像调大电力按钮一样简单。

灵活性 & 峰值处理能力

在访问量剧增的情况下,应用仍然需要继续发挥作用,但是这样的突发流量并不常见;如果为以能处理这类峰值访问为标准来投入资源随时待命无疑是巨大的浪费。使用消息队列能够使关键组件顶住突发的访问压力,而不会因为突发的超负荷的请求而完全崩溃。

可恢复性

当体系的一部分组件失效,不会影响到整个系统。消息队列降低了进程间的耦合度,所以即使一个处理消息的进程挂掉,加入队列中的消息仍然可以在系统恢复后被处理。而这种允许重试或者延后处理请求的能力通常是造就一个略感不便的用户和一个沮丧透顶的用户之间的区别。

送达保证

消息队列提供的冗余机制保证了消息能被实际的处理,只要一个进程读取了该队列即可。在此基础上, IronMQ 提供了一个”只送达一次”保证。无论有多少进程在从队列中领取数据,每一个消息只能被处理一次。这之所以成为可能,是因为获取一个消息只是”预定”了这个消息,暂时把它移出了队列。除非客户端明确的表示已经处理完了这个消息,否则这个消息会被放回队列中去,在一段可配置的时间之后可再次被处理。

顺序保证

在大多使用场景下,数据处理的顺序都很重要。消息队列本来就是排序的,并且能保证数据会按照特定的顺序来处理。 IronMO 保证消息通过 FIFO(先进先出)的顺序来处理,因此消息在队列中的位置就是从队列中检索他们的位置。

缓冲

在任何重要的系统中,都会有需要不同的处理时间的元素。例如,加载一张图片比应用过滤器花费更少的时间。消息队列通过一个缓冲层来帮助任务最高效率的执行—写入队列的处理会尽可能的快速,而不受从队列读的预备处理的约束。该缓冲有助于控制和优化数据流经过系统的速度。

理解数据流

在一个分布式系统里,要得到一个关于用户操作会用多长时间及其原因的总体印象,是个巨大的挑战。消息系列通过消息被处理的频率,来方便的辅助确定那些表现不佳的处理过程或领域,这些地方的数据流都不够优化。

异步通信

很多时候,你不想也不需要立即处理消息。消息队列提供了异步处理机制,允许你把一个消息放入队列,但并不立即处理它。你想向队列中放入多少消息就放多少,然后在你乐意的时候再去处理它们。

Looper

Handler、Looper、Message、MessageQueue关系是什么

HandleLooper(轮循器)、MessageQueue(消息队列)、Message(消息)


Message:Handler接收和处理的消息对象。

MessageQueue:消息队列,他采用先进先出的方式来管理Message。Handler发送消息时先把消息发送到消息队列中。

Looper:每个线程最多拥有一个Looper。他的Loop方法负责读取MessageQueue队列中的消息,读到信息之后就把信息交给发送该消息的Handler进行处理。程序在创建Looper对象的时候,会自动在Looper的构造器中自动创建一个MessageQueue队列对象。

注意:Handler发送的消息时先放入MessageQueue,所以handler要正常工作,必须当前线程先有MessageQueue

而MessageQueue是由Looper管理的,所以换句话说,Handler要正常工作必须先有Looper。

为了保证线程中有Looper,分两种情况。

总结:


主线程中使用Handler直接创建即可以使用。

子线程中使用Handler的步骤:

a:调用Loop.prepare();

b:创建Handler对象;

c:调用Looper.loop();


目录
相关文章
|
24天前
|
消息中间件 存储 Java
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
|
27天前
|
消息中间件 存储 Java
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
47 1
|
1月前
|
消息中间件 存储 Java
Android消息处理机制(Handler+Looper+Message+MessageQueue)
Android消息处理机制(Handler+Looper+Message+MessageQueue)
45 2
|
5天前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。