Android HandlerThread 總結使用

来源:http://www.cnblogs.com/zhaoyanjun/archive/2016/11/14/6062880.html
-Advertisement-
Play Games

轉載請標明出處:http://www.cnblogs.com/zhaoyanjun/p/6062880.html 本文出自 "【趙彥軍的博客】" 前言 以前我在 "【Android Handler、Loop 的簡單使用】" 介紹了子線程和子線程之間的通信。 很明顯的一點就是,我們要在子線程中調用Lo ...


轉載請標明出處:http://www.cnblogs.com/zhaoyanjun/p/6062880.html
本文出自【趙彥軍的博客】

前言

以前我在 【Android Handler、Loop 的簡單使用】 介紹了子線程和子線程之間的通信。
這裡寫圖片描述

很明顯的一點就是,我們要在子線程中調用Looper.prepare() 為一個線程開啟一個消息迴圈,預設情況下Android中新誕生的線程是沒有開啟消息迴圈的。(主線程除外,主線程系統會自動為其創建Looper對象,開啟消息迴圈。) Looper對象通過MessageQueue來存放消息和事件。一個線程只能有一個Looper,對應一個MessageQueue。 然後通過Looper.loop() 讓Looper開始工作,從消息隊列里取消息,處理消息。

註意:寫在Looper.loop()之後的代碼不會被執行,這個函數內部應該是一個迴圈,當調用mHandler.getLooper().quit()後,loop才會中止,其後的代碼才能得以運行。

然而這一切都可以用HandlerThread類來幫我們做這些邏輯操作。

HandlerThread

HandlerThread本質上就是一個普通Thread,只不過內部建立了Looper.

HandlerThread的常規用法

  • 創建一個HandlerThread

mThread = new HandlerThread("handler_thread");

  • 啟動一個HandlerThread

    mThread.start();

  • 退出迴圈
    Looper是通過調用loop方法驅動著消息迴圈的進行: 從MessageQueue中阻塞式地取出一個消息,然後讓Handler處理該消息,周而複始,loop方法是個死迴圈方法。

那如何終止消息迴圈呢?我們可以調用Looper的quit方法或quitSafely方法,二者稍有不同。

 /**
     * Quits the looper.
     * <p>
     * Causes the {@link #loop} method to terminate without processing any
     * more messages in the message queue.
     * </p><p>
     * Any attempt to post messages to the queue after the looper is asked to quit will fail.
     * For example, the {@link Handler#sendMessage(Message)} method will return false.
     * </p><p class="note">
     * Using this method may be unsafe because some messages may not be delivered
     * before the looper terminates.  Consider using {@link #quitSafely} instead to ensure
     * that all pending work is completed in an orderly manner.
     * </p>
     *
     * @see #quitSafely
     */
    public void quit() {
        mQueue.quit(false);
    }

    /**
     * Quits the looper safely.
     * <p>
     * Causes the {@link #loop} method to terminate as soon as all remaining messages
     * in the message queue that are already due to be delivered have been handled.
     * However pending delayed messages with due times in the future will not be
     * delivered before the loop terminates.
     * </p><p>
     * Any attempt to post messages to the queue after the looper is asked to quit will fail.
     * For example, the {@link Handler#sendMessage(Message)} method will return false.
     * </p>
     */
    public void quitSafely() {
        mQueue.quit(true);
    }

相同點:
將不在接受新的事件加入消息隊列。

不同點
當我們調用Looper的quit方法時,實際上執行了MessageQueue中的removeAllMessagesLocked方法,該方法的作用是把MessageQueue消息池中所有的消息全部清空,無論是延遲消息(延遲消息是指通過sendMessageDelayed或通過postDelayed等方法發送的需要延遲執行的消息)還是非延遲消息。

當我們調用Looper的quitSafely方法時,實際上執行了MessageQueue中的removeAllFutureMessagesLocked方法,通過名字就可以看出,該方法只會清空MessageQueue消息池中所有的延遲消息,並將消息池中所有的非延遲消息派發出去讓Handler去處理,quitSafely相比於quit方法安全之處在於清空消息之前會派發所有的非延遲消息。

無論是調用了quit方法還是quitSafely方法只會,Looper就不再接收新的消息。即在調用了Looper的quit或quitSafely方法之後,消息迴圈就終結了,這時候再通過Handler調用sendMessage或post等方法發送消息時均返回false,表示消息沒有成功放入消息隊列MessageQueue中,因為消息隊列已經退出了。

需要註意的是Looper的quit方法從API Level 1就存在了,但是Looper的quitSafely方法從API Level 18才添加進來。

小例子:

package com.app;

import android.os.Bundle;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;

public class MainActivity extends AppCompatActivity {

    private HandlerThread myHandlerThread ;
    private Handler handler ;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        //創建一個線程,線程名字:handler-thread
        myHandlerThread = new HandlerThread( "handler-thread") ;
        //開啟一個線程
        myHandlerThread.start();
        //在這個線程中創建一個handler對象
        handler = new Handler( myHandlerThread.getLooper() ){
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
                //這個方法是運行在 handler-thread 線程中的 ,可以執行耗時操作
                Log.d( "handler " , "消息: " + msg.what + "  線程: " + Thread.currentThread().getName()  ) ;

            }
        };

        //在主線程給handler發送消息
        handler.sendEmptyMessage( 1 ) ;

        new Thread(new Runnable() {
            @Override
            public void run() {
             //在子線程給handler發送數據
             handler.sendEmptyMessage( 2 ) ;
            }
        }).start() ;

    }

    @Override
    protected void onDestroy() {
        super.onDestroy();

        //釋放資源
        myHandlerThread.quit() ;
    }
}

運行效果:

/com.app D/handler: 消息: 1  線程: handler-thread
/com.app D/handler: 消息: 2  線程: handler-thread

HandlerThread的特點

  • HandlerThread將loop轉到子線程中處理,說白了就是將分擔MainLooper的工作量,降低了主線程的壓力,使主界面更流暢。

  • 開啟一個線程起到多個線程的作用。處理任務是串列執行,按消息發送順序進行處理。HandlerThread本質是一個線程,線上程內部,代碼是串列處理的。

  • 但是由於每一個任務都將以隊列的方式逐個被執行到,一旦隊列中有某個任務執行時間過長,那麼就會導致後續的任務都會被延遲處理。

  • HandlerThread擁有自己的消息隊列,它不會幹擾或阻塞UI線程。

  • 對於網路IO操作,HandlerThread並不適合,因為它只有一個線程,還得排隊一個一個等著。

參考文章:

Android Handler、Loop 的簡單使用
Android 如何有效的解決記憶體泄漏的問題
Android 更新UI的幾種方式


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

-Advertisement-
Play Games
更多相關文章
  • 本地預覽圖片html和js例子,直接上代碼吧。 <!--WizRtf2Html Charset=0 --><!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <meta http-equiv="Content-Type ...
  • 1、前端基礎 盒子模型 位置(相對 絕對) 瀏覽器渲染順序(從上而下,從外到里;如果裡面有寬度則再從里而在,如div沒有設置寬度則一直撐大到設置了寬度的div為止) 浮動 2、基礎佈局 中間固定兩邊自適應(就是居中了) 可用display:inline-block 和text-align:cente ...
  • 在CSS3中提供了對邊框進行圓角設定的支持,可對邊框1~4個角進行圓角樣式設置。 ...
  • 1、SVG繪圖總結: ①方法一:已有svg文件,<img src="x.svg"> 方法二:<body><svg></svg></body> ②繪製矩形:<rect x="" y="" width="" height=""></rect> ③繪製圓形:<circle cx="" cy="" r="" ...
  • jquery美化選擇器實例有:邊框、下劃線、 伸縮 、滑動、 覆蓋、 旋轉、 彈出層選擇 、環形效果。 線上預覽 實例代碼 <body class="demo-1"> <div class="container"> <header class="codrops-header"> <h1> 自定義選擇 ...
  • //創建新的tcp伺服器var net = require('net');var chatServer = net.createServer()chatServer.on('connection',function(client){ client.write('Hi\n'); client.writ ...
  • 先上效果圖: 獲取手機已安裝的App列表利用Android系統API就可以辦到,這裡為什麼要引入RxJava?現在我們假設一下有下麵幾個需求: 1、我們不需要所有的App,只需要用戶安裝的第三方App,即過濾到系統App; 2、我們自定義一個AppInfo類,該類中保存了App_Icon、App_N ...
  • 今天學到了這個Loader,淺談一下自己的看法: 1.定義 Loader是一個載入器,可以用來它訪問數據,可以看做訪問數據的機器(好比挖掘機)。裝再器從android3.0開始引進,它使得在activity或fragment中非同步載入數據變得簡單。 具有如下特別: 1)它們對每個Activity和F ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...