Android訊息機制(2)

來源:互聯網
上載者:User

標籤:android   io   使用   ar   for   strong   資料   sp   div   

在Android 中,線程內部或者線程之間進行資訊互動時經常會使用訊息,這些基礎的東西如果我們熟悉其內部的原理,將會使我們容易、更好地架構系統,避免一些低級的錯誤。

下面我們分析下程式的運行過程:

1.onCreate()

首先啟動服務時將會調用onCreate()方法,在該方法中我們new了一個HandlerThread對象,提供了線程的名字和優先順序。

緊接著我們調用了start()方法,執行該方法將會調用HandlerThread對象的run()方法:

  1. public void run() {      
  2.         mTid = Process.myTid();      
  3.         Looper.prepare();      
  4.         synchronized (this) {      
  5.             mLooper = Looper.myLooper();      
  6.             notifyAll();      
  7.         }      
  8.         Process.setThreadPriority(mPriority);      
  9.         onLooperPrepared();      
  10.         Looper.loop();      
  11.         mTid = -1;      
  12.     }  

在run()方法中,系統給線程添加的Looper,同時調用了Looper的loop()方法:

  1. public static final void loop() {        
  2.         Looper me = myLooper();      
  3.         MessageQueue queue = me.mQueue;      
  4.         while (true) {      
  5.             Message msg = queue.next(); // might block      
  6.             //if (!me.mRun) {      
  7.             //    break;      
  8.             //}      
  9.             if (msg != null) {      
  10.                 if (msg.target == null) {      
  11.                     // No target is a magic identifier for the quit message.      
  12.                     return;      
  13.                 }      
  14.                 if (me.mLogging!= null) me.mLogging.println(      
  15.                         ">>>>> Dispatching to " + msg.target + " "     
  16.                         + msg.callback + ": " + msg.what      
  17.                         );      
  18.                 msg.target.dispatchMessage(msg);      
  19.                 if (me.mLogging!= null) me.mLogging.println(      
  20.                         "<<<<< Finished to    " + msg.target + " "     
  21.                         + msg.callback);      
  22.                 msg.recycle();      
  23.             }      
  24.         }      
  25.     }   

通過源碼我們可以看到loop()方法是個死迴圈,將會不停的從MessageQueue對象中擷取Message對象,如果 MessageQueue 對象中不存在Message對象,則結束本次迴圈,然後繼續迴圈;如果存在Message對象,則執行 msg.target.dispatchMessage(msg),但是這個msg的.target欄位的值是什麼呢?我們先暫時停止跟蹤源碼,返回到 onCreate()方法中。線程執行完start()方法後,我們可以擷取線程的Looper對象,然後new一個ServiceHandler對象, 我們把Looper對象傳到ServiceHandler建構函式中將使handler、looper和messageQueue三者建立聯絡。

2.onStartCommand()

執行完onStart()方法後,將執行onStartCommand()方法。首先我們從訊息池中擷取一個Message執行個體,然後給 Message對象的arg1、what、obj三個欄位賦值。緊接著調用sendMessage(msg)方法,我們跟蹤原始碼,該方法將會調用 sendMessageDelayed(msg, 0)方法,而sendMessageDelayed()方法又會調用sendMessageAtTime(msg, SystemClock.uptimeMillis() + delayMillis)方法,在該方法中我們要注意該句代碼msg.target = this,msg的target指向了this,而this就是ServiceHandler對象,因此msg的target欄位指向了 ServiceHandler對象,同時該方法又調用MessageQueue 的enqueueMessage(msg, uptimeMillis)方法:

  1. final boolean enqueueMessage(Message msg, long when) {      
  2.         if (msg.when != 0) {      
  3.             throw new AndroidRuntimeException(msg      
  4.                     + " This message is already in use.");      
  5.         }      
  6.         if (msg.target == null && !mQuitAllowed) {      
  7.             throw new RuntimeException("Main thread not allowed to quit");      
  8.         }      
  9.         synchronized (this) {      
  10.             if (mQuiting) {      
  11.                 RuntimeException e = new RuntimeException(      
  12.                     msg.target + " sending message to a Handler on a dead thread");      
  13.                 Log.w("MessageQueue", e.getMessage(), e);      
  14.                 return false;      
  15.             } else if (msg.target == null) {      
  16.                 mQuiting = true;      
  17.             }      
  18.             msg.when = when;      
  19.             //Log.d("MessageQueue", "Enqueing: " + msg);      
  20.             Message p = mMessages;      
  21.             if (p == null || when == 0 || when < p.when) {      
  22.                 msg.next = p;      
  23.                 mMessages = msg;      
  24.                 this.notify();      
  25.             } else {      
  26.                 Message prev = null;      
  27.                 while (p != null && p.when <= when) {      
  28.                     pprev = p;      
  29.                     pp = p.next;      
  30.                 }      
  31.                 msg.next = prev.next;      
  32.                 prev.next = msg;      
  33.                 this.notify();      
  34.             }      
  35.         }      
  36.         return true;      
  37.     }   

該方法主要的任務就是把Message對象的添加到MessageQueue中(資料結構最基礎的東西,自己畫圖理解下)。

handler.sendMessage()-->handler.sendMessageDelayed()-->handler.sendMessageAtTime()-->msg.target = this;queue.enqueueMessage==>把msg添加到訊息佇列中

3.handleMessage(msg)

onStartCommand()執行完畢後我們的Service中的方法就執行完畢了,那麼handleMessage()是怎麼調用的呢?在前 面分析的loop()方法中,我們當時不知道msg的target欄位代碼什麼,通過上面分析現在我們知道它代表ServiceHandler對 象,msg.target.dispatchMessage(msg);則表示執行ServiceHandler對象中的 dispatchMessage()方法

  1. public void dispatchMessage(Message msg) {      
  2.         if (msg.callback != null) {      
  3.             handleCallback(msg);      
  4.         } else {      
  5.             if (mCallback != null) {      
  6.                 if (mCallback.handleMessage(msg)) {      
  7.                     return;      
  8.                 }      
  9.             }      
  10.             handleMessage(msg);      
  11.         }      
  12.     }  

該方法首先判斷callback是否為空白,我們跟蹤的過程中未見給其賦值,因此callback欄位為空白,所以最終將會執行handleMessage()方法,也就是我們ServiceHandler類中複寫的方法。在該方法將根據what欄位的值判斷執行哪段代碼。

至此,我們看到,一個Message經由Handler的發送,MessageQueue的入隊,Looper的抽取,又再一次地回到Handler的懷抱中。而繞的這一圈,也正好協助我們將同步操作變成了非同步作業。

Android訊息機制(2)

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.