Android應用程序消息處理機制(Looper、Handler)分析(4)
管道是Linux系統(tǒng)中的一種進程間通信機制,具體可以參考前面一篇文章Android學習啟動篇推薦的一本書《Linux內(nèi)核源代碼情景分析》中的第6 章--傳統(tǒng)的Uinx進程間通信。
簡單來說,管道就是一個文件,在管道的兩端,分別是兩個打開文件文件描述符,這兩個打開文件描述符都是對應同一個文件, 其中一個是用來讀的,別一個是用來寫的,一般的使用方式就是,一個線程通過讀文件描述符中來讀管道的內(nèi)容,當管道沒有內(nèi)容時,這個線程就會進入等待狀態(tài), 而另外一個線程通過寫文件描述符來向管道中寫入內(nèi)容,寫入內(nèi)容的時候,如果另一端正有線程正在等待管道中的內(nèi)容,那么這個線程就會被喚醒。
這個等待和喚醒 的操作是如何進行的呢,這就要借助Linux系統(tǒng)中的epoll機制了。
Linux系統(tǒng)中的epoll機制為處理大批量句柄而作了改進的poll,是Linux下多路復用IO接口select/poll的增強版本,它能顯著減 少程序在大量并發(fā)連接中只有少量活躍的情況下的系統(tǒng)CPU利用率。
但是這里我們其實只需要監(jiān)控的IO接口只有mWakeReadPipeFd一個,即前面我們所創(chuàng)建的管道的讀端,為什么還需要用到epoll呢?有點用牛刀來殺雞的味道。
其實不然,這個Looper類是非常強大的,它除了監(jiān)控內(nèi)部所創(chuàng)建的管 道接口之外,還提供了addFd接口供外界面調(diào)用,外界可以通過這個接口把自己想要監(jiān)控的IO事件一并加入到這個Looper對象中去,當所有這些被監(jiān)控 的IO接口上面有事件發(fā)生時,就會喚醒相應的線程來處理,不過這里我們只關(guān)心剛才所創(chuàng)建的管道的IO事件的發(fā)生。
我們再回到NativeMessageQueue的構(gòu)造函數(shù)中,看看JNI層的Looper對象的創(chuàng)建過程,即看看它的構(gòu)造函數(shù)是如何實現(xiàn)的,這個Looper類實現(xiàn)在frameworks/base/libs/utils/Looper.cpp文件中:
- [cpp] view plaincopyLooper::Looper(bool allowNonCallbacks) :
- mAllowNonCallbacks(allowNonCallbacks),
- mResponseIndex(0) {
- int wakeFds[2];
- int result = pipe(wakeFds);
- ......
- mWakeReadPipeFd = wakeFds[0];
- mWakeWritePipeFd = wakeFds[1];
- ......
- #ifdef LOOPER_USES_EPOLL
- // Allocate the epoll instance and register the wake pipe.
- mEpollFd = epoll_create(EPOLL_SIZE_HINT);
- ......
- struct epoll_event eventItem;
- memset(& eventItem, 0, sizeof(epoll_event)); // zero out unused members
- data field union
- eventItem.events = EPOLLIN;
- eventItem.data.fd = mWakeReadPipeFd;
- result = epoll_ctl(mEpollFd, EPOLL_CTL_ADD, mWakeReadPipeFd, &
- entItem);
- ......
- #else
- ......
- #endif
- ......
- }
這個構(gòu)造函數(shù)做的事情非常重要,它跟我們后面要介紹的應用程序主線程在消息隊列中沒有消息時要進入等待狀態(tài)以及當消息隊列有消息時要把應用程序主線程喚醒的這兩個知識點息息相關(guān)。它主要就是通過pipe系統(tǒng)調(diào)用來創(chuàng)建了一個管道了:
- [cpp] view plaincopyint wakeFds[2];
- int result = pipe(wakeFds);
- ......
- mWakeReadPipeFd = wakeFds[0];
- mWakeWritePipeFd = wakeFds[1];
要使用Linux系統(tǒng)的epoll機制,首先要通過epoll_create來創(chuàng)建一個epoll專用的文件描述符:
- [cpp] view plaincopymEpollFd = epoll_create(EPOLL_SIZE_HINT);
傳入的參數(shù)EPOLL_SIZE_HINT是在這個mEpollFd上能監(jiān)控的***文件描述符數(shù)。