Looper、Hander、HandlerThread
一.Message 、Looper、Handler之间的关系
1.系统发送的Message消息传送给Handler,Handler将Message放入自己的looper队列的底部
然后再从Looper的顶部获取Message信息,并执行方法。
2.一个Handler只有一个Looper,一个Message仅与一个Handler交互,多个Handler可以与一个Looper交互。
3.主线程自带Hanlder和Looper
4.小技巧:为避免创建Message对象,所以调用Handler.obtainMessage()方法从公共池中获取Message,发送message:message.sendToTarget()。
//假设已经有一个Handler mHandler.obtainMessage(int what,Obj obj) .sendToTarget();
Message的解剖
Message最重要的三个变量:
what:用来描述信息,相当于Map的键值对的键。//因为有很多message发送给Handler所以Handler需要分辨
obj:存储对象,相当于Map的键值对的值。//Msg发送给Handler的对象
target:处理消息的Handler
arg1,arg2:存储数字,准备用来发送给Handler
Message在准备处理状态下,Handler负责处理行为。
注:通常对Message对象不是直接new出来的,只要调用handler中的obtainMessage方法来直接获得Message对象。
Looper的解剖
1.Looper的作用:是负责管理消息队列,负责消息的出列和入列操作。
2.流程:系统发送的Message消息传送给Handler,Handler将Message放入自己的looper队列的底部,然后再从Looper的顶部获取Message信息。
一个Handler只有一个Looper,多个Handler可以与一个Looper交互。
3.Looper的使用:Looper.myLooper():获取当前进程的looper对象,类似的 Looper.getMainLooper() 用于获取主线程的Looper对象。
Looper.prepare():创建Looper(主线程自带Looper,在子线程中使用创建Looper)
Looper.loop(); 让Looper开始工作,从消息队列里取消息,处理消息。
注意:写在Looper.loop()之后的代码不会被执行,这个函数内部应该是一个循环,当调用mHandler.getLooper().quit()后,loop才会中止,其后的代码才能得以运行。
Handler的解剖
1.Handler主要有两个用途:首先是可以定时处理或者分发消息,其次是可以添加一个执行的行为在其它线程中执行
2.使用:子类需要继承Hendler类,并重写handleMessage(Message msg) 方法, 用于接受线程数据。
创建Handler的三种方法
//直接创建匿名类 缺点:默认为主进程的Looper Handler mHandler = new Handler() { @Override public void handleMessage(Message msg) {//覆盖handleMessage方法 switch (msg.what) {//根据收到的消息的what类型处理 case BUMP_MSG: Log.v("handler", "Handler===="+msg.arg1);//打印收到的消息 break; default: super.handleMessage(msg);//这里最好对不需要或者不关心的消息抛给父类,避免丢失消息 break; } } //用new Handler(Looper looper,CallBack) Handler myHandler = new Handler(new Callback(){ // 参数也可以为(this.getMainLooper(),new Callback(){})不写则默认为主线程的Looper,如果是在子线程中创建,Looper默认为子线程的Looper
//Handler不带参数的默认构造函数:new Handler(),实际上是通过Looper.myLooper()来获取当前线程中的消息循环,详见3 @Override public boolean handleMessage(Message msg) { // TODO Auto-generated method stub return false; } }); //子类继承Handler class MyHandler extends Handler { public MyHandler() { } //默认为主线程的Looper public MyHandler(Looper L) { super(L); } //自定义的Looper // 子类必须重写此方法,接受数据 @Override public void handleMessage(Message msg) { // TODO Auto-generated method stub Log。d("MyHandler", "handleMessage。。。。。。"); super。handleMessage(msg); // 此处可以更新UI Bundle b = msg。getData(); String color = b。getString("color"); MyHandlerActivity。this。button。append(color); } }
3.在非主线程中Looper与Handler和Thread
在非主线程中直接new Handler() 错误,原因是非主线程中默认没有创建Looper对象,需要先调用Looper.prepare()启用Looper。
默认情况下,线程是没有消息循环的,所以要调用 Looper.prepare()来给线程创建消息循环,然后再通过,Looper.loop()来使消息循环起作用。
class LooperThread extends Thread { public Handler mHandler; public void run() { Looper.prepare(); mHandler = new Handler() { public void handleMessage(Message msg) { // process incoming messages here } }; Looper.loop(); } }
作用:可实现主线程给子线程(非主线程)发送消息
注:UI最好只能在主线程中运行。
二、HandlerThread
简介:继承自Thread,线程同时创建了一个含有消息队列的Looper,并对外提供自己这个Looper对象的get方法,这就是它和普通Thread唯一不同的地方。
使用:
-
创建一个
HandlerThread
,即创建了一个包含Looper的线程。HandlerThread handlerThread = new HandlerThread("leochin.com");
handlerThread.start(); //创建HandlerThread后一定要记得start()
-
获取
HandlerThread
的Looper(getLooper()是在start()之后使用的)Looper looper = handlerThread.getLooper();
-
创建Handler,通过Looper初始化
Handler handler = new Handler(looper);
通过以上三步我们就成功创建HandlerThread
。通过handler发送消息,就会在子线程中执行。
4. 关闭HandlerThread
handlerThread.quit();
子类继承HandlerThread时创建的小技巧:
class MyHandlerThread extends HandlerThread {
public MyHandlerThread(String name) {
super(name);
}//一定要定义当前Thread的名字
}
2.onLooperPrepare()是最适合创建内部Handler的地方