真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

Android中如何利用Input子系統(tǒng)監(jiān)聽線程的啟動(dòng)

這篇文章主要講解了“Android中如何利用Input子系統(tǒng)監(jiān)聽線程的啟動(dòng)”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“Android中如何利用Input子系統(tǒng)監(jiān)聽線程的啟動(dòng)”吧!

專注于為中小企業(yè)提供網(wǎng)站設(shè)計(jì)制作、成都網(wǎng)站建設(shè)服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)邯山免費(fèi)做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動(dòng)了1000+企業(yè)的穩(wěn)健成長(zhǎng),幫助中小企業(yè)通過(guò)網(wǎng)站建設(shè)實(shí)現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。

InputManagerService初始化概覽

首先,有幾點(diǎn)共識(shí)我們都可以達(dá)成:

  • Android  Framework層的Service(Java)都是由system_server進(jìn)程創(chuàng)建的(由于沒有fork,因此都運(yùn)行在system_server進(jìn)程中)

  • Service創(chuàng)建后就會(huì)交給運(yùn)行在system_server進(jìn)程中的ServiceManager管理。

因此對(duì)于InputManagerService的創(chuàng)建,我們可以在SystemServer的startOtherServices()方法中找到,該方法做了以下事情:

  • 創(chuàng)建InputManagerService對(duì)象

  • 將它交給ServiceManager管理

  • 將WindowManagerService的InputMonitor注冊(cè)到InputManagerService中作為窗口響應(yīng)事件后的回調(diào)

  • 完成以上工作后啟動(dòng)InputManagerService。 

SystemServer.javastartOtherServices(){     ……     inputManager = new InputManagerService(context);     ……     inputManager.setWindowManagerCallbacks(wm.getInputMonitor());     inputManager.start();     …… }

接下來(lái)我們就逐部分學(xué)習(xí)相應(yīng)的處理。

InputManagerService對(duì)象的創(chuàng)建

創(chuàng)建InputManagerService對(duì)象時(shí)會(huì)完成以下工作:

  • 創(chuàng)建一個(gè)負(fù)責(zé)處理DisplayThread線程中的Message的Handler

  • 調(diào)用nativeInit初始化native層的InputManagerService,初始化的時(shí)候傳入了DisplayThread的消息隊(duì)列

  • 用mPtr保存native層的InputManagerService

  • 初始化完成后將Service添加到LocalServices,通過(guò)Map以鍵值對(duì)的形式存儲(chǔ) 

InputManagerService.javapublic InputManagerService(Context context) {    this.mContext = context;    this.mHandler = new InputManagerHandler(DisplayThread.get().getLooper());      mUseDevInputEventForAudioJack =             context.getResources().getBoolean(R.bool.config_useDevInputEventForAudioJack);     Slog.i(TAG, "Initializing input manager, mUseDevInputEventForAudioJack="             + mUseDevInputEventForAudioJack);     mPtr = nativeInit(this, mContext, mHandler.getLooper().getQueue());      LocalServices.addService(InputManagerInternal.class, new LocalService()); }

這里可能有人就會(huì)問了,為什么InputManagerService要和DisplayThread綁定在一起?大家不妨想想,InputEvent無(wú)論如何被獲取、歸類、分發(fā),最終還是要被處理,也就意味著最終它的處理結(jié)果都要在UI上體現(xiàn),那么InputManagerService自然要選擇和UI親近一些的線程在一起了。

但是問題又來(lái)了,應(yīng)用都是運(yùn)行在自己的主線程里的,難道InputManagerService要一個(gè)個(gè)綁定么,還是一個(gè)個(gè)輪詢?這些做法都太過(guò)低效,那換個(gè)辦法,可不可以和某個(gè)管理或非常親近所有應(yīng)用UI的線程綁定在一起呢?

答案是什么,我在這里先不說(shuō),大家可以利用自己的知識(shí)想想。

初始化native層的InputManagerService

在nativeInit函數(shù)中,將Java層的MessageQueue轉(zhuǎn)換為native層的MessageQueue,然后再取出Looper用于NativeInputManager的初始化??梢娺@里的重頭戲就是NativeInputManager的創(chuàng)建,這個(gè)過(guò)程做了以下事情:

  • 將Java層的Context和InputManagerService轉(zhuǎn)換為native層的Context和InputManagerService存儲(chǔ)在mContextObj和mServiceObj中

  • 初始化變量

  • 創(chuàng)建EventHub

  • 創(chuàng)建InputManager 

com_android_server_input_InputManagerService.cpp  NativeInputManager::NativeInputManager(jobject contextObj,         jobject serviceObj, const sp& looper) :         mLooper(looper), mInteractive(true) {     JNIEnv* env = jniEnv();      mContextObj = env->NewGlobalRef(contextObj);     mServiceObj = env->NewGlobalRef(serviceObj);      {        AutoMutex _l(mLock);         mLocked.systemUiVisibility = ASYSTEM_UI_VISIBILITY_STATUS_BAR_VISIBLE;         mLocked.pointerSpeed = 0;         mLocked.pointerGesturesEnabled = true;         mLocked.showTouches = false;     }     mInteractive = true;      sp eventHub = new EventHub();     mInputManager = new InputManager(eventHub, this, this); }

EventHub

看到這里很多人就會(huì)想,EventHub是什么?取英語(yǔ)釋義來(lái)看,它的意思是事件樞紐。我們?cè)谖恼麻_頭的時(shí)候也提到過(guò),Input系統(tǒng)的事件來(lái)源于驅(qū)動(dòng)/內(nèi)核,那么我們可以猜測(cè)EventHub是處理來(lái)自驅(qū)動(dòng)/內(nèi)核的元事件的樞紐。接下來(lái)就在源碼中驗(yàn)證我們的想法吧。

EventHub的創(chuàng)建過(guò)程中做了以下事情:

  • 創(chuàng)建mEpollFd用于監(jiān)聽是否有數(shù)據(jù)(有無(wú)事件)可讀

  • 創(chuàng)建mINotifyFd將它注冊(cè)到DEVICE_PATH(這里路徑就是/dev/input)節(jié)點(diǎn),并將它交給內(nèi)核用于監(jiān)聽該設(shè)備節(jié)點(diǎn)的增刪數(shù)據(jù)事件。那么只要有數(shù)據(jù)增刪的事件到來(lái),epoll_wait()就會(huì)返回,使得EventHub能收到來(lái)自系統(tǒng)的通知,并獲取事件的詳細(xì)信息

  • 調(diào)用epoll_ctl函數(shù)將mEpollFd和mINotifyFd注冊(cè)到epoll中

  • 定義int wakeFd[2]作為事件傳輸管道的讀寫兩端,并將讀端注冊(cè)到epoll中讓mEpollFd監(jiān)聽 

EventHub.cpp  EventHub::EventHub(void) :         mBuiltInKeyboardId(NO_BUILT_IN_KEYBOARD), mNextDeviceId(1), mControllerNumbers(),         mOpeningDevices(0), mClosingDevices(0),         mNeedToSendFinishedDeviceScan(false),         mNeedToReopenDevices(false), mNeedToScanDevices(true),         mPendingEventCount(0), mPendingEventIndex(0), mPendingINotify(false) {     acquire_wake_lock(PARTIAL_WAKE_LOCK, WAKE_LOCK_ID);      mEpollFd = epoll_create(EPOLL_SIZE_HINT);     LOG_ALWAYS_FATAL_IF(mEpollFd < 0, "Could not create epoll instance.  errno=%d", errno);      mINotifyFd = inotify_init();     int result = inotify_add_watch(mINotifyFd, DEVICE_PATH, IN_DELETE | IN_CREATE);     ……     result = epoll_ctl(mEpollFd, EPOLL_CTL_ADD, mINotifyFd, &eventItem);     ……     int wakeFds[2];     result = pipe(wakeFds);     ……     mWakeReadPipeFd = wakeFds[0];     mWakeWritePipeFd = wakeFds[1];      result = fcntl(mWakeReadPipeFd, F_SETFL, O_NONBLOCK);     ……     result = fcntl(mWakeWritePipeFd, F_SETFL, O_NONBLOCK);     ……     result = epoll_ctl(mEpollFd, EPOLL_CTL_ADD, mWakeReadPipeFd, &eventItem);     …… }

那么這里拋出一個(gè)問題:為什么要把管道的讀端注冊(cè)到epoll中?假如EventHub因?yàn)間etEvents讀不到事件而阻塞在epoll_wait()里,而我們沒有綁定讀端的話,我們要怎么喚醒EventHub?如果綁定了管道的讀端,我們就可以通過(guò)向管道的寫端寫數(shù)據(jù)從而讓EventHub因?yàn)榈玫焦艿缹懚说臄?shù)據(jù)而被喚醒。

InputManager的創(chuàng)建

接下來(lái)繼續(xù)說(shuō)InputManager的創(chuàng)建,它的創(chuàng)建就簡(jiǎn)單多了,創(chuàng)建一個(gè)InputDispatcher對(duì)象用于分發(fā)事件,一個(gè)InputReader對(duì)象用于讀事件并把事件交給InputDispatcher分發(fā),,然后調(diào)用initialize()初始化,其實(shí)也就是創(chuàng)建了InputReaderThread和InputDispatcherThread。

InputManager.cpp  InputManager::InputManager(        const sp& eventHub,        const sp& readerPolicy,        const sp& dispatcherPolicy) {     mDispatcher = new InputDispatcher(dispatcherPolicy);     mReader = new InputReader(eventHub, readerPolicy, mDispatcher);     initialize(); }void InputManager::initialize() {     mReaderThread = new InputReaderThread(mReader);     mDispatcherThread = new InputDispatcherThread(mDispatcher); }

InputDispatcher和InputReader的創(chuàng)建都相對(duì)簡(jiǎn)單。InputDispatcher會(huì)創(chuàng)建自己線程的Looper,以及設(shè)置根據(jù)傳入的dispatchPolicy設(shè)置分發(fā)規(guī)則。InputReader則會(huì)將傳入的InputDispatcher封裝為監(jiān)聽對(duì)象存起來(lái),做一些數(shù)據(jù)初始化就結(jié)束了。

至此,InputManagerService對(duì)象的初始化就完成了,根據(jù)開頭說(shuō)的,接下來(lái)就會(huì)調(diào)用InputManagerService的start()方法。

監(jiān)聽線程InputReader和InputDispatcher的啟動(dòng)

在start()方法中,做了以下事情:

  • 調(diào)用nativeStart方法,其實(shí)就是調(diào)用InputManager的start()方法

  • 將InputManagerService交給WatchDog監(jiān)控

  • 注冊(cè)觸控點(diǎn)速度、顯示觸控的觀察者,并注冊(cè)廣播監(jiān)控它們

  • 主動(dòng)調(diào)用updateXXX方法更新(初始化) 

InputManagerService.javapublic void start() {     Slog.i(TAG, "Starting input manager");     nativeStart(mPtr);    // Add ourself to the Watchdog monitors.     Watchdog.getInstance().addMonitor(this);      registerPointerSpeedSettingObserver();     registerShowTouchesSettingObserver();     registerAccessibilityLargePointerSettingObserver();      mContext.registerReceiver(new BroadcastReceiver() {        @Override         public void onReceive(Context context, Intent intent) {             updatePointerSpeedFromSettings();             updateShowTouchesFromSettings();             updateAccessibilityLargePointerFromSettings();         }     }, new IntentFilter(Intent.ACTION_USER_SWITCHED), null, mHandler);      updatePointerSpeedFromSettings();     updateShowTouchesFromSettings();     updateAccessibilityLargePointerFromSettings(); }

顯而易見這里最值得關(guān)注的就是InputManager的start()方法了,可惜這個(gè)方法并不值得我們?nèi)绱岁P(guān)心,因?yàn)樗龅氖虑楹芎?jiǎn)單,就是啟動(dòng)InputDispatcherThread和InputReaderThread開始監(jiān)聽。

status_t InputManager::start() {     status_t result = mDispatcherThread->run("InputDispatcher", PRIORITY_URGENT_DISPLAY);    if (result) {         ALOGE("Could not start InputDispatcher thread due to error %d.", result);        return result;     }      result = mReaderThread->run("InputReader", PRIORITY_URGENT_DISPLAY);    if (result) {         ALOGE("Could not start InputReader thread due to error %d.", result);          mDispatcherThread->requestExit();        return result;     }    return OK; }

那么InputReaderThread線程是怎么和EventHub關(guān)聯(lián)起來(lái)的呢?

對(duì)于InputReadThread:

  • 啟動(dòng)后循環(huán)執(zhí)行mReader->loopOnce()

  • loopOnce()中會(huì)調(diào)用mEventHub->getEvents讀取事件

  • 讀到了事件就會(huì)調(diào)用processEventsLocked處理事件

  • 處理完成后調(diào)用getInputDevicesLocked獲取輸入設(shè)備信息

  • 調(diào)用mPolicy->notifyInputDevicesChanged函數(shù)利用InputManagerService的代理通過(guò)Handler發(fā)送MSG_DELIVER_INPUT_DEVICES_CHANGED消息,通知輸入設(shè)備發(fā)生了變化

  • ***調(diào)用mQueuedListener->flush(),將事件隊(duì)列中的所有事件交給在InputReader中注冊(cè)過(guò)的InputDispatcher 

bool InputReaderThread::threadLoop() {     mReader->loopOnce();    return true; }void InputReader::loopOnce() {     ……      size_t count = mEventHub->getEvents(timeoutMillis, mEventBuffer, EVENT_BUFFER_SIZE);      { // acquire lock         AutoMutex _l(mLock);         mReaderIsAliveCondition.broadcast();        if (count) {             processEventsLocked(mEventBuffer, count);         }      ……        if (oldGeneration != mGeneration) {             inputDevicesChanged = true;             getInputDevicesLocked(inputDevices);         }     } // release lock      // Send out a message that the describes the changed input devices.     if (inputDevicesChanged) {         mPolicy->notifyInputDevicesChanged(inputDevices);     }      ……     mQueuedListener->flush(); }

感謝各位的閱讀,以上就是“Android中如何利用Input子系統(tǒng)監(jiān)聽線程的啟動(dòng)”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)Android中如何利用Input子系統(tǒng)監(jiān)聽線程的啟動(dòng)這一問題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!


文章標(biāo)題:Android中如何利用Input子系統(tǒng)監(jiān)聽線程的啟動(dòng)
文章分享:http://weahome.cn/article/gihhsh.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部