Android HandlerThread案例詳解
HandlerThread 顧名思義就是一種可以使用 Handler 的 Thread。日常開發中我們經常會通過創建一個 Thread 去執行任務,有多個任務就多創建幾個線程實現,這時候可能出現線程同步的問題。不過有時候我們并不需要很強的并發性,只需保證按照順序地執行各個任務即可,有什么好辦法實現呢?第一反應想到的可能是通過 Executors.newSingleThreadExecutor() 方法來創建一個 SingleThreadExecutor,來統一所有的任務到一個線程中,然后按順序執行。其實,除了這個方法之外,HandlerThread 也可以實現。
簡單使用首先創建一個 HandlerThreadActivity
public class HandlerThreadActivity extends BaseActivity {private static final String TAG = 'HandlerThreadActivity'; private Button mStartBtn; private Handler mHandler; private HandlerThread mHandlerThread; @Override protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_handler_thread);mStartBtn = findViewById(R.id.start_btn);mHandlerThread = new HandlerThread('THREAD_NAME');mHandlerThread.start();mHandler = new Handler(mHandlerThread.getLooper());mStartBtn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {mHandler.post(new Runnable() { @Override public void run() {Log.d(TAG, Thread.currentThread().getId() + ' ' + String.valueOf((Looper.myLooper() == Looper.getMainLooper())) + ' 任務:' + this.hashCode());SystemClock.sleep(3000); }}); }}); } @Override protected void onDestroy() {super.onDestroy();mHandlerThread.quit(); }}
快速三擊按鈕,打印日志如下:
可以發現,三次不同的任務按開始的順序執行,而且是運行在子線程中,那到底是怎么實現的呢?
源碼解析public class HandlerThread extends Thread { int mPriority; int mTid = -1; Looper mLooper; private @Nullable Handler mHandler; public HandlerThread(String name) {super(name);mPriority = Process.THREAD_PRIORITY_DEFAULT; } public HandlerThread(String name, int priority) {super(name);mPriority = priority; } protected void onLooperPrepared() { } @Override public void run() {// 獲取線程 idmTid = Process.myTid();//構建一個 LooperLooper.prepare();synchronized (this) { mLooper = Looper.myLooper(); notifyAll();}//設置線程優先級Process.setThreadPriority(mPriority);onLooperPrepared();Looper.loop();// Looper 循環mTid = -1; } // 獲取當前線程的 Looper, public Looper getLooper() {if (!isAlive()) { return null;} synchronized (this) { while (isAlive() && mLooper == null) {try { wait();} catch (InterruptedException e) {} }}return mLooper; } /** * @return a shared {@link Handler} associated with this thread * @hide 方法隱藏掉,無法調用 */ @NonNull public Handler getThreadHandler() {if (mHandler == null) { mHandler = new Handler(getLooper());}return mHandler; } //線程退出方法,主要是調用 Looper.quit() 方法,不然一直在循環 public boolean quit() {Looper looper = getLooper();if (looper != null) { looper.quit(); return true;}return false; } //同上,不過這個方法會把消息隊列中的已有消息處理完才會安全地退出 public boolean quitSafely() {Looper looper = getLooper();if (looper != null) { looper.quitSafely(); return true;}return false; } public int getThreadId() {return mTid; }}
通讀下來,如果熟悉 Handler 原理的同學大概就明白 HandlerThread 的機制了:
HandlerThread 運行 start() 方法,回調 run() 方法。 在 run() 方法中通過 Looper.prepare() 來創建消息隊列,并通過 Looper.looper() 方法來開啟消息循環。 由于 Loop.loop() 是一個死循環,導致 run() 也是無線循環,因此當我們不需要使用 HandlerThread 的時候,要調用它的 quit() 方法或者 quiteSafely() 方法。到此這篇關于Android HandlerThread案例詳解的文章就介紹到這了,更多相關Android HandlerThread內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!
相關文章:
1. php測試程序運行速度和頁面執行速度的代碼2. ASP中常用的22個FSO文件操作函數整理3. 三個不常見的 HTML5 實用新特性簡介4. Warning: require(): open_basedir restriction in effect,目錄配置open_basedir報錯問題分析5. ASP調用WebService轉化成JSON數據,附json.min.asp6. SharePoint Server 2019新特性介紹7. React+umi+typeScript創建項目的過程8. 無線標記語言(WML)基礎之WMLScript 基礎第1/2頁9. ASP.NET Core 5.0中的Host.CreateDefaultBuilder執行過程解析10. php網絡安全中命令執行漏洞的產生及本質探究
