Android應(yīng)用程序消息處理機制(Looper、Handler)分析(5)
這里就是告訴mEpollFd,它要監(jiān)控mWakeReadPipeFd文件描述符的EPOLLIN事件,即當管道中有內(nèi)容可讀時,就喚醒當前正在等待管道中的內(nèi)容的線程。
C++層的這個Looper對象創(chuàng)建好了之后,就返回到JNI層的NativeMessageQueue的構(gòu)造函數(shù),***就返回到Java層的消息 隊 列MessageQueue的創(chuàng)建過程,這樣,Java層的Looper對象就準備好了。有點復(fù)雜,我們先小結(jié)一下這一步都做了些什么事情:
A. 在Java層,創(chuàng)建了一個Looper對象,這個Looper對象是用來進入消息循環(huán)的,它的內(nèi)部有一個消息隊列MessageQueue對象mQueue;
B. 在JNI層,創(chuàng)建了一個NativeMessageQueue對象,這個NativeMessageQueue對象保存在Java層的消息隊列對象mQueue的成員變量mPtr中;
C. 在C++層,創(chuàng)建了一個Looper對象,保存在JNI層的NativeMessageQueue對象的成員變量mLooper中,這個對象的作用是,當 Java層的消息隊列中沒有消息時,就使Android應(yīng)用程序主線程進入等待狀態(tài),而當Java層的消息隊列中來了新的消息后,就喚醒Android應(yīng) 用程序的主線程來處理這個消息。
接著還要通過epoll_ctl函數(shù)來告訴epoll要監(jiān)控相應(yīng)的文件描述符的什么事件:
- [cpp] view plaincopystruct epoll_event eventItem;
- memset(& eventItem, 0, sizeof(epoll_event)); // zero out unused members
- of data field union
- eventItem.events = EPOLLIN;
- eventItem.data.fd = mWakeReadPipeFd;
- result = epoll_ctl(mEpollFd, EPOLL_CTL_ADD, mWakeReadPipeFd, &
- eventItem);
回到ActivityThread類的main函數(shù)中,在上面這些工作都準備好之后,就調(diào)用Looper類的loop函數(shù)進入到消息循環(huán)中去了:
- [cpp] view plaincopypublic class Looper {
- ......
- public static final void loop() {
- Looper me = myLooper();
- MessageQueue queue = me.mQueue;
- ......
- while (true) {
- Message msg = queue.next(); // might block
- ......
- if (msg != null) {
- if (msg.target == null) {
- // No target is a magic identifier for the quit message.
- return;
- }
- ......
- msg.target.dispatchMessage(msg);
- ......
- msg.recycle();
- }
- }
- }
- ......
- }
這里就是進入到消息循環(huán)中去了,它不斷地從消息隊列mQueue中去獲取下一個要處理的消息msg,如果消息的target成員變量為null,就表示要 退出消息循環(huán)了,否則的話就要調(diào)用這個target對象的dispatchMessage成員函數(shù)來處理這個消息,這個target對象的類型為 Handler,下面我們分析消息的發(fā)送時會看到這個消息對象msg是如設(shè)置的。