Android線程管理——線程通信

来源:http://www.cnblogs.com/younghao/archive/2016/01/13/5116819.html
-Advertisement-
Play Games

線程通信、ActivityThread及Thread類是理解Android線程管理的關鍵。 線程,作為CPU調度資源的基本單位,在Android等針對嵌入式設備的操作系統中,有著非常重要和基礎的作用。本小節主要從以下三個方面進行分析: 《Android線程管理——線程通信》 《Android線程管理...


      線程通信、ActivityThread及Thread類是理解Android線程管理的關鍵。

      線程,作為CPU調度資源的基本單位,在Android等針對嵌入式設備的操作系統中,有著非常重要和基礎的作用。本小節主要從以下三個方面進行分析:

  1. 《Android線程管理——線程通信》
  2. 《Android線程管理——ActivityThread》
  3. 《Android線程管理——Thread》


一、Handler、MessageQueue、Message及Looper四者的關係

      在開發Android多線程應用時,Handler、MessageQueue、Message及Looper是老生常談的話題。但想徹底理清它們之間的關係,卻需要深入的研究下它們各自的實現才行。首先,給出一張它們之間的關係圖:

繪圖1

  • Looper依賴於MessageQueue和Thread,因為每個Thread只對應一個Looper,每個Looper只對應一個MessageQueue。
  • MessageQueue依賴於Message,每個MessageQueue對應多個Message。即Message被壓入MessageQueue中,形成一個Message集合。
  • Message依賴於Handler進行處理,且每個Message最多指定一個Handler來處理。Handler依賴於MessageQueue、Looper及Callback。

      從運行機制來看,Handler將Message壓入MessageQueue,Looper不斷從MessageQueue中取出Message(當MessageQueue為空時,進入休眠狀態),其target handler則進行消息處理。因此,要徹底弄清Android的線程通信機制,需要瞭解以下三個問題:

  • Handler的消息分發、處理流程
  • MessageQueue的屬性及操作
  • Looper的工作原理

1.1 Handler的消息分發、處理流程

      Handler主要完成Message的入隊(MessageQueue)和處理,下麵將通過Handler的源碼分析其消息分發、處理流程。首先,來看下Handler類的方法列表:

2697.tm

     從上圖中可以看出,Handler類核心的方法包括:1)構造器;2)分發消息;3)處理消息;4)post發送消息;5)send發送消息;6)remove消息和回調。

     首先,從構造方法來看,構造器的多態最終通過調用如下方法實現,即將實參賦值給Handler類的內部域。

final MessageQueue mQueue;
final Looper mLooper;
final Callback mCallback;
final boolean mAsynchronous;

public Handler(Looper looper, Callback callback, boolean async) {
    mLooper = looper;
    mQueue = looper.mQueue;
    mCallback = callback;
    mAsynchronous = async;
}

     其次,消息的入隊是通過post方法和send方法來實現的。

public final boolean postAtTime(Runnable r, long uptimeMillis) {
    return sendMessageAtTime(getPostMessage(r), uptimeMillis);
}
public final boolean sendEmptyMessageAtTime(int what, long uptimeMillis) {
    Message msg = Message.obtain();
    msg.what = what;
    return sendMessageAtTime(msg, uptimeMillis);
}
public boolean sendMessageAtTime(Message msg, long uptimeMillis) {
    MessageQueue queue = mQueue;
    if (queue == null) {
        RuntimeException e = new RuntimeException(
                this + " sendMessageAtTime() called with no mQueue");
        Log.w("Looper", e.getMessage(), e);
        return false;
    }
    return enqueueMessage(queue, msg, uptimeMillis);
}

      兩者的區別在於參數類型不同,post方法傳入的實例對象實現了Runnable介面,然後在內部通過getPostMessage方法將其轉換為Message,最終通過send方法發出;send方法傳入的實例對象為Message類型,在實現中,將Message壓入MessageQueue。

private static Message getPostMessage(Runnable r) {
    Message m = Message.obtain();
    m.callback = r;
    return m;
}

      通過Handler將Message壓入MessageQueue之後,Looper將其輪詢後交由Message的target handler處理。Handler首先會對消息進行分發。首先判斷Message的回調處理介面Callback是否為null,不為null則調用該Callback進行處理;否判斷Handler的回調介面mCallback是否為null,不為null則調用該Callback進行處理;如果上述Callback均為null,則調用handleMessage方法處理。

public void dispatchMessage(Message msg) {
    if (msg.callback != null) {
        handleCallback(msg);
    } else {
        if (mCallback != null) {
            if (mCallback.handleMessage(msg)) {
                return;
            }
        }
        handleMessage(msg);
    }
}

      handleMessage方法在Handler的子類中必須實現。即消息具體的處理交由應用軟體實現。

/**
 * Subclasses must implement this to receive messages.
 */
public void handleMessage(Message msg) {
}

      回到Activity(Fragment),在Handler的子類中實現handleMessage方法。這裡需要註意一個記憶體泄露的問題,比較下述兩種實現方式,第一種直接定義Handler的實現,第二種通過靜態內部類繼承Handler,定義繼承類的實例。

Handler mHandler = new Handler() {
            
    @Override
    public void handleMessage(Message msg) {
        super.handleMessage(msg);
            
        // 根據msg調用Activity的方法
    }
};
static class MyHandler extends Handler {

    WeakReference<DemoActivity> mActivity;

    public MyHandler(DemoActivity demoActivity) {
        mActivity = new WeakReference<DemoActivity>(demoActivity);
    }

    @Override
    public void handleMessage(Message msg) {
        super.handleMessage(msg);
        DemoActivity theActivity = mActivity.get();

        // 根據msg調用theActivity的方法
}

      不繞彎子,直接說明為什麼第一種方式會引起記憶體泄露,而第二種不會。

      在第一種方式中,mHandler通過匿名內部類方式實例化,在Java中,內部類會強持有外部類的引用(handleMessage方法中可以直接調用Activity的方法),在外部Activity調用onDestroy()方法之後,如果Handler的MessageQueue依然有未處理的消息,那麼由於Handler持有Activity的引用導致Activity無法被系統GC回收,從而引起記憶體泄露。

      在第二種方式中,首先繼承Handler定義靜態內部類,由於MyHandler為靜態類,即使定義在Activity的內部,也與Activity沒有邏輯上的聯繫,即不會持有外部Activity的引用;其次,在靜態類內部,定義外部Activity的弱引用,弱引用在系統資源緊張時會被系統優先回收。最後,在handleMessage()方法中,通過WeakReference的get方法獲取外部Activity的引用,如果該弱引用已被回收,則get方法返回null。

struct GcSpec {
  /* If true, only the application heap is threatened. */
  bool isPartial;
  /* If true, the trace is run concurrently with the mutator. */
  bool isConcurrent;
  /* Toggles for the soft reference clearing policy. */
  bool doPreserve;
  /* A name for this garbage collection mode. */
  const char *reason;
};

      這段代碼定義在dalvik/vm/alloc/Heap.h中,其中doPreserve為true時,表示在執行GC的過程中,不回收軟引用引用的對象;為false時,表示在執行GC的過程中,回收軟引用引用的對象。

      最後,使用Handler的過程中,還需要註意一點,在前面的方法列表圖中已經提到。為避免Activity調用onDestroy後,Handler的MessageQueue中仍存在Message,一般會在onDestroy中調用removeCallbacksAndMessages()方法。

@Override
protected void onDestroy() {
    super.onDestroy();
    // 清空Message隊列
    myHandler.removeCallbacksAndMessages(null);
}
public final void removeCallbacksAndMessages(Object token) {
    mQueue.removeCallbacksAndMessages(this, token);
}

      removeCallbacksAndMessages()方法會移除obj為token的由post發送的callback和send發送的message,當token為null時,會移除所有callback和message。

1.2 MessageQueue的屬性及操作

      MessageQueue,消息隊列,其屬性與常規隊列相似,包括入隊、出隊等,這裡簡要介紹一下MessageQueue的實現。

      首先,MessageQueue新建隊列的工作是通過在其構造器中調用本地方法nativeInit實現的。nativeInit會創建NativeMessageQueue對象,然後賦值給MessageQueue成員變數mPtr。mPtr是int類型數據,代表NativeMessageQueue的記憶體指針。

MessageQueue(boolean quitAllowed) {
    mQuitAllowed = quitAllowed;
    mPtr = nativeInit();
}

      其次,Message入隊的通過enqueueMessage方法實現。首先檢查message是否符合入隊要求(是否正在使用,target handler是否為null),符合要求後通過設置prev.next = msg隊列的指針完成入隊操作。

boolean enqueueMessage(Message msg, long when);

      再次,出隊是通過next()方法完成的。涉及到同步、鎖等問題,這裡不詳細展開了。

      再次,刪除元素有兩個實現。即分別通過p.callback == r和p.what == what來進行消息識別。

void removeMessages(Handler h, int what, Object object);
void removeMessages(Handler h, Runnable r, Object object);

      最後,銷毀隊列和創建隊列一樣,是通過本地函數完成的。傳入的參數為MessageQueue的記憶體指針。

private native static void nativeDestroy(int ptr);

1.3 Looper的工作原理

      Looper是線程通信的關鍵,正是因為Looper,整個線程通信機制才真正實現“通”。

      在應用開發過程中,一般當主線程需要傳遞消息給用戶自定義線程時,會在自定義線程中定義Handler進行消息處理,併在Handler實現的前後分別調用Looper的prepare()方法和loop()方法。大致實現如下:

new Thread(new Runnable() {
            
    private Handler mHandler;
            
    @Override
    public void run() {
        Looper.prepare();
        mHandler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
                        
            }
        };
        Looper.loop();
    }
});

這裡重點說明prepare()方法和loop()方法,實際項目中不建議定義匿名線程。

 

private static void prepare(boolean quitAllowed) {
    if (sThreadLocal.get() != null) {
        throw new RuntimeException("Only one Looper may be created per thread");
    }
    sThreadLocal.set(new Looper(quitAllowed));
}

      可以看出,prepare方法的重點是sThreadLocal變數,sThreadLocal變數是什麼呢?

// sThreadLocal.get() will return null unless you've called prepare().
static final ThreadLocal<Looper> sThreadLocal = new ThreadLocal<Looper>();

      ThreadLocal實現了線程本地存儲。簡單看一下它的類註解文檔,ThreadLocal是一種特殊的全局變數,全局性在於它存儲於自己所線上程相關的數據,而其他線程無法訪問。

/**
 * Implements a thread-local storage, that is, a variable for which each thread
 * has its own value. All threads share the same {@code ThreadLocal} object,
 * but each sees a different value when accessing it, and changes made by one
 * thread do not affect the other threads. The implementation supports
 * {@code null} values.
 *
 * @see java.lang.Thread
 * @author Bob Lee
 */
public class ThreadLocal<T> {
}

      回到prepare方法中,sThreadLocal添加了一個針對當前線程的Looper對象。並且prepare方法只能調用一次,否則會拋出運行時異常。

      初始化完畢之後,Handler通過post和send方法如何保證消息投遞到Looper所持有的MessageQueue中呢?其實,MessageQueue是Handler和Looper的橋梁。在前面Handler章節中提到Handler的初始化方法,Handler的mLooper對象是通過Looper的靜態方法myLooper()獲取的,而myLooper()是通過調用sThreadLocal.get()來得到的,即Handler的mLooper就是當前線程的Looper對象,Handler的mQueue就是mLooper.mQueue。

……
mLooper = Looper.myLooper();
if (mLooper == null) {
   throw new RuntimeException(
        "Can't create handler inside thread that has not called Looper.prepare()");
}
mQueue = mLooper.mQueue;
……
public static Looper myLooper() {
    return sThreadLocal.get();
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Adapter(適配器的講解)適配器就我自己來看,我覺得這是一個非常重要的知識點,Adapter是用來幫助填出數據的中間橋梁,簡單點說吧:將各種數據以合適的形式顯示在View中給用戶看。Adapter有很多的介面、抽象類、子類可以使用,這裡就我們常用的幾個進行講解BaseAdapter,ArrayA...
  • 1、EditPlus下載地址http://rj.baidu.com/soft/detail/12876.html?ald激活碼:http://www.jb51.net/tools/editplus/
  • // 該代碼在網上找的視頻中的例子,感覺很適合類和對象分不清楚的同學參考,僅供學習分享,謝謝// 創建一個Pointtest類,用屬性x、y表示點的坐標位置,求兩點之間的距離,使用兩種方法:類方法和對象方法#import #import // 要使用到開方和求平方根兩個函數,pow和sqrt,所以....
  • 現在網上的有很多人寫單例模式,一個很基本的東西但是版本也有很多,新人看了難免有些眼花繚亂的感覺。自己最新比較閑,也過來寫一些自己的心得。在往下看之前,我們要明白一點,那就是在什麼情況下我們才要用到單例模式呢?單例模式在一般情況下用於當一個類只能有一個實例的時候,或者說當一個類只需要定義一個,而且還要...
  • 原文鏈接:http://www.cnblogs.com/liufan9/archive/2013/04/02/2995626.html對於以前做C#或者JAVA開發的朋友而言,初次接觸iOS開發,Obj-C的函數命名方式可能會感覺非常不習慣。尤其是打開AppDelegate.m,映入眼帘的代碼竟然是...
  • 在軟體開發過程中,我們常常需要在某個時間後執行某個方法,或者是按照某個周期一直執行某個方法。在這個時候,我們就需要用到定時器。然而,在iOS中有很多方法完成以上的任務,到底有多少種方法呢?經過查閱資料,大概有三種方法:NSTimer、CADisplayLink、GCD。接下來我就一一介紹它們的用法。...
  • nil、Nil和NSNull的理解 nil和null從字面意思來理解比較簡單,nil是一個對象,而NULL是一個值,我的理解為nil是將對象設置為空,而null是將基本類型設置為空的,個人感覺有點像屬性當中,基本類型分配為assign NSString類型一般分配copy,而對象一般用retai.....
  • 前言:方法替換,可以替換任意外部類的方法,而動態添加方法只能實現在被添加類創建的對象里,但是將方法替換和動態添加方法結合使用,可以實現,對任意外部類動態添加需要的方法,這個方法可以是類方法也可以是實例方法,這個外部類也可以是沒有任何方法聲明和實現的類。主要思路:使用運行時的方法替換將在外部類將自定義...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...