Android消息机--handler的自我总结
大家都知道Handler是Android为了能在子线程更新UI所创作出来的一个类.
那么为什么Handler能在handler里面能更新UI。是因为Handler 把更新UI的操作切换到了主线程来操作.
有些博客说Handler是线程之间的一种通讯方式,这只局限于子线程跟主线程之间,
那么子线程是怎么把消息传送到主线程来操作,这其中一个关键的地方就是Looper.
从例子我们可以知道 假如现在有一段主要的代码
oncrete(){
new Thread(new Runnable(){
run{
//进行UI操作 ---这样是会报错的 那么我们加上handler
// 之前是没有的,现在加上Looper
Looper.prepare();
new Handler(){
handlerMessage(){
//进行UI操作 ---
// 如果是这样依然会报错 因为子线程中现在还没有Looper,会报没有Looper那个错误信息(从源码可以看到加了Looper检测) 那么我们加上Looper
}
}
Looper.loop();
}
}).start;
}
从上面的代码中 加了Looper之后还是会报错,报应该在主线程中执行UI操作.
那么是为什么,原因在于 这个Looper不是主线程的Looper
一个解决方法是 把Looper.prepare();改成Looper.getMainLooper();
或者 new Handler(Looper.getMainLooper);
这样就可以更新UI不会报错了,
那么Looper是怎样把更新UI的操作切换到主线程
我们从源码可以看到 当我们new Handler(Looper.getMainLooper())的时候 里面有
Handler(LooperlooperCallbackcallbackasync){=looper=looper.mQueue=callback=async}
所以说这个时候 handler已经拿到了主线程的Looper;
然后我们在Looper的loop();方法里面看到
{msg.target.dispatchMessage(msg)}{(traceTag!=){Trace.(traceTag)}}
会调用msg.target 这个msg.target是在这里拿到
Looperme=()(me==){RuntimeException()}MessageQueuequeue=me.mQueue
(){Messagemsg=queue.next()(msg==){}
在loop()方法里面Looper会开始无限循环 去处理信息啦。
顺序是以下这样
1:实例化Activity 的时候创建ActivityThread(主线程) 然后在Main()里面创建Looper 然后Looper里面通过mThreadLocal 的set方法保存主线程
然后再调用Looper.loop();这时候会开启无限循环
2:Looper.getMainLooper 会调用get方法 这个时候就拿到了主线程的Looper 以及里面保存的messageQueue
然后 handler调用post 其实也就是在主线程的messageQueue上插入一个信息
那么loop 方法里面就检测到了,这时候已经是切换到主进程了.
看源码比较能清晰整个流程
声明:本站所有文章资源内容,如无特殊说明或标注,均为采集网络资源。如若本站内容侵犯了原著者的合法权益,可联系本站删除。