欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

android子線程,Android子線程向主線程傳遞消息

android創(chuàng)建子線程

你是要怎么個意思?

公司主營業(yè)務(wù):成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)、移動網(wǎng)站開發(fā)等業(yè)務(wù)。幫助企業(yè)客戶真正實(shí)現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競爭能力。創(chuàng)新互聯(lián)是一支青春激揚(yáng)、勤奮敬業(yè)、活力青春激揚(yáng)、勤奮敬業(yè)、活力澎湃、和諧高效的團(tuán)隊(duì)。公司秉承以“開放、自由、嚴(yán)謹(jǐn)、自律”為核心的企業(yè)文化,感謝他們對我們的高要求,感謝他們從不同領(lǐng)域給我們帶來的挑戰(zhàn),讓我們激情的團(tuán)隊(duì)有機(jī)會用頭腦與智慧不斷的給客戶帶來驚喜。創(chuàng)新互聯(lián)推出秀英免費(fèi)做網(wǎng)站回饋大家。

創(chuàng)建子線程即就是在你的程序體(也就是MainThread)中創(chuàng)建子線程:

1、匿名內(nèi)部類對Thread類進(jìn)行覆寫:new

Thread(){

覆寫run方法

}.start();

2、或者

new

Thread(new

Runnable(){········}).start();套用兩層匿名內(nèi)部類也可以

3、再或者就是將當(dāng)前類繼承Runnable然后在當(dāng)前類里覆寫上run方法,最后在需要開線程的地方寫上new

Thread(this).start();

####別忘了在更新UI的時候跳回主線程就行了,用runOnUiThread

和handle機(jī)制都可以####

Android線程池的使用

在Android中有主線程和子線程的區(qū)分。主線程又稱為UI線程,主要是處理一些和界面相關(guān)的事情,而子線程主要是用于處理一些耗時比較大的一些任務(wù),例如一些網(wǎng)絡(luò)操作,IO請求等。如果在主線程中處理這些耗時的任務(wù),則有可能會出現(xiàn)ANR現(xiàn)象(App直接卡死)。

線程池,從名字的表明含義上我們知道線程池就是包含線程的一個池子,它起到新建線程、管理線程、調(diào)度線程等作用。

既然Android中已經(jīng)有了線程的概念,那么為什么需要使用線程池呢?我們從兩個方面給出使用線程池的原因。

在Android中線程池就是ThreadPoolExecutor對象。我們先來看一下ThreadPoolExecutor的構(gòu)造函數(shù)。

我們分別說一下當(dāng)前的幾個參數(shù)的含義:

第一個參數(shù)corePoolSize為 核心線程數(shù) ,也就是說線程池中至少有這么多的線程,即使存在的這些線程沒有執(zhí)行任務(wù)。但是有一個例外就是,如果在線程池中設(shè)置了allowCoreThreadTimeOut為true,那么在 超時時間(keepAliveTime) 到達(dá)后核心線程也會被銷毀。

第二個參數(shù)maximumPoolSize為 線程池中的最大線程數(shù) 。當(dāng)活動線程數(shù)達(dá)到這個數(shù)后,后續(xù)添加的新任務(wù)會被阻塞。

第三個參數(shù)keepAliveTime為 線程的保活時間 ,就是說如果線程池中有多于核心線程數(shù)的線程,那么在線程沒有任務(wù)的那一刻起開始計(jì)時,如果超過了keepAliveTime,還沒有新的任務(wù)過來,則該線程就要被銷毀。同時如果設(shè)置了allowCoreThreadTimeOut為true,該時間也就是上面第一條所說的 超時時間 。

第四個參數(shù)unit為 第三個參數(shù)的計(jì)時單位 ,有毫秒、秒等。

第五個參數(shù)workQueue為 線程池中的任務(wù)隊(duì)列 ,該隊(duì)列持有由execute方法傳遞過來的Runnable對象(Runnable對象就是一個任務(wù))。這個任務(wù)隊(duì)列的類型是BlockQueue類型,也就是阻塞隊(duì)列,當(dāng)隊(duì)列的任務(wù)數(shù)為0時,取任務(wù)的操作會被阻塞;當(dāng)隊(duì)列的任務(wù)數(shù)滿了(活動線程達(dá)到了最大線程數(shù)),添加操作就會阻塞。

第六個參數(shù)threadFactory為 線程工廠 ,當(dāng)線程池需要創(chuàng)建一個新線程時,使用線程工廠來給線程池提供一個線程。

第七個參數(shù)handler為 拒絕策略 ,當(dāng)線程池使用有界隊(duì)列時(也就是第五個參數(shù)),如果隊(duì)列滿了,任務(wù)添加到線程池的時候的一個拒絕策略。

可以看到FixedThreadPool的構(gòu)建調(diào)用了ThreadPoolExecutor的構(gòu)造函數(shù)。從上面的調(diào)用中可以看出FixedThreadPool的幾個特點(diǎn):

可以看到CacheThreadPool的構(gòu)建調(diào)用了ThreadPoolExecutor的構(gòu)造函數(shù)。從上面的調(diào)用中可以看出CacheThreadPool的幾個特點(diǎn):

可以看到ScheduledThreadPoolExecutor的構(gòu)建調(diào)用了ThreadPoolExecutor的構(gòu)造函數(shù)。從上面的調(diào)用中可以看出ScheduledThreadPoolExecutor的幾個特點(diǎn):

可以看到SingleThreadExecutor的構(gòu)建調(diào)用了ThreadPoolExecutor的構(gòu)造函數(shù)。從上面的調(diào)用中可以看出SingleThreadExecutor的幾個特點(diǎn):

Android中的特殊線程——HandlerThread

一般我們想要執(zhí)行耗時操作都會想開啟Thread子線程去處理,但是多次創(chuàng)建和銷毀線程是很耗系統(tǒng)資源的。為了解決這一問題,我們可以自己構(gòu)建一個循環(huán)線程,在線程當(dāng)中創(chuàng)建一個Looper輪循器,來進(jìn)行消息的輪循, 當(dāng)有耗時任務(wù)需要投放到該循環(huán)線程時,線程就會執(zhí)行耗時任務(wù),任務(wù)執(zhí)行完成之后,線程又會處于阻塞等待狀態(tài),不會馬上銷毀掉,直到下一個耗時任務(wù)被投放進(jìn)來;即通過阻塞和等待來保證性能最優(yōu),為此Google為我們封裝好了HandlerThread框架。

HandlerThread本質(zhì)上就是一個Thread子線程,不同的是在Thread內(nèi)部有一個Looper開啟了輪循器;

HandlerThread = Handler + Thread + Looper

由于一般的Thread沒有開啟Looper輪循器,所以不能在子線程中創(chuàng)建handler,因?yàn)闆]有對應(yīng)的MessageQueue與handler相關(guān)聯(lián),而MessageQueue是由Looper進(jìn)行維護(hù)的;如果想在子線程中創(chuàng)建handler,必須先通過Looper.prepare()創(chuàng)建一個Looper,再通過Looper.loop()開啟循環(huán),才能使用Handler。

1、HandlerThread本質(zhì)上是一個線程類,繼承了Thread;

2、HandlerThread有自己的內(nèi)部Looper對象,可以進(jìn)行l(wèi)oop循環(huán);所以在HandlerThread中可以創(chuàng)建handler來發(fā)送和處理消息;

3、通過獲取HandlerThread的Looper對象傳遞給Handler,可以在handleMessage()中執(zhí)行異步操作;

4、handler中的looper默認(rèn)綁定了UI線程的MessageQueue,對于非UI線程想使用MessageQueue機(jī)制的,HandlerThread的內(nèi)部Looper最適合,它不會干擾和阻塞UI線程,減少了對性能的消耗,但是處理效率低;

5、HandlerThread是一個串行隊(duì)列,會比較穩(wěn)定;

Android 中的“子線程”解析

Android 中線程可分為 主線程 和 子線程 兩類,其中主線程也就是 UI線程 ,它的主要這作用就是運(yùn)行四大組件、處理界面交互。子線程則主要是處理耗時任務(wù),也是我們要重點(diǎn)分析的。

首先 Java 中的各種線程在 Android 里是通用的,Android 特有的線程形態(tài)也是基于 Java 的實(shí)現(xiàn)的,所以有必要先簡單的了解下 Java 中的線程,本文主要包括以下內(nèi)容:

在 Java 中要創(chuàng)建子線程可以直接繼承 Thread 類,重寫 run() 方法:

或者實(shí)現(xiàn) Runnable 接口,然后用Thread執(zhí)行Runnable,這種方式比較常用:

簡單的總結(jié)下:

Callable 和 Runnable 類似,都可以用來處理具體的耗時任務(wù)邏輯的,但是但具體的差別在哪里呢?看一個小例子:

定義 MyCallable 實(shí)現(xiàn)了 Callable 接口,和之前 Runnable 的 run() 方法對比下, call() 方法是有返回值的哦,泛型就是返回值的類型:

一般會通過線程池來執(zhí)行 Callable (線程池相關(guān)內(nèi)容后邊會講到),執(zhí)行結(jié)果就是一個 Future 對象:

可以看到,通過線程池執(zhí)行 MyCallable 對象返回了一個 Future 對象,取出執(zhí)行結(jié)果。

Future 是一個接口,從其內(nèi)部的方法可以看出它提供了取消任務(wù)(有坑!!!)、判斷任務(wù)是否完成、獲取任務(wù)結(jié)果的功能:

Future 接口有一個 FutureTask 實(shí)現(xiàn)類,同時 FutureTask 也實(shí)現(xiàn)了 Runnable 接口,并提供了兩個構(gòu)造函數(shù):

用 FutureTask 一個參數(shù)的構(gòu)造函數(shù)來改造下上邊的例子:

FutureTask 內(nèi)部有一個 done() 方法,代表 Callable 中的任務(wù)已經(jīng)結(jié)束,可以用來獲取執(zhí)行結(jié)果:

所以 Future + Callable 的組合可以更方便的獲取子線程任務(wù)的執(zhí)行結(jié)果,更好的控制任務(wù)的執(zhí)行,主要的用法先說這么多了,其實(shí) AsyncTask 內(nèi)部也是類似的實(shí)現(xiàn)!

注意, Future 并不能取消掉運(yùn)行中的任務(wù),這點(diǎn)在后邊的 AsyncTask 解析中有提到。

Java 中線程池的具體的實(shí)現(xiàn)類是 ThreadPoolExecutor ,繼承了 Executor 接口,這些線程池在 Android 中也是通用的。使用線程池的好處:

常用的構(gòu)造函數(shù)如下:

一個常規(guī)線程池可以按照如下方式來實(shí)現(xiàn):

執(zhí)行任務(wù):

基于 ThreadPoolExecutor ,系統(tǒng)擴(kuò)展了幾類具有新特性的線程池:

線程池可以通過 execute() 、 submit() 方法開始執(zhí)行任務(wù),主要差別從方法的聲明就可以看出,由于 submit() 有返回值,可以方便得到任務(wù)的執(zhí)行結(jié)果:

要關(guān)閉線程池可以使用如下方法:

IntentService 是 Android 中一種特殊的 Service,可用于執(zhí)行后臺耗時任務(wù),任務(wù)結(jié)束時會自動停止,由于屬于系統(tǒng)的四大組件之一,相比一般線程具有較高的優(yōu)先級,不容易被殺死。用法和普通 Service 基本一致,只需要在 onHandleIntent() 中處理耗時任務(wù)即可:

至于 HandlerThread,它是 IntentService 內(nèi)部實(shí)現(xiàn)的重要部分,細(xì)節(jié)內(nèi)容會在 IntentService 源碼中說到。

IntentService 首次創(chuàng)建被啟動的時候其生命周期方法 onCreate() 會先被調(diào)用,所以我們從這個方法開始分析:

這里出現(xiàn)了 HandlerThread 和 ServiceHandler 兩個類,先搞明白它們的作用,以便后續(xù)的分析。

首先看 HandlerThread 的核心實(shí)現(xiàn):

首先它繼承了 Thread 類,可以當(dāng)做子線程來使用,并在 run() 方法中創(chuàng)建了一個消息循環(huán)系統(tǒng)、開啟消息循環(huán)。

ServiceHandler 是 IntentService 的內(nèi)部類,繼承了 Handler,具體內(nèi)容后續(xù)分析:

現(xiàn)在回過頭來看 onCreate() 方法主要是一些初始化的操作, 首先創(chuàng)建了一個 thread 對象,并啟動線程,然后用其內(nèi)部的 Looper 對象 創(chuàng)建一個 mServiceHandler 對象,將子線程的 Looper 和 ServiceHandler 建立了綁定關(guān)系,這樣就可以使用 mServiceHandler 將消息發(fā)送到子線程去處理了。

生命周期方法 onStartCommand() 方法會在 IntentService 每次被啟動時調(diào)用,一般會這里處理啟動 IntentService 傳遞 Intent 解析攜帶的數(shù)據(jù):

又調(diào)用了 start() 方法:

就是用 mServiceHandler 發(fā)送了一條包含 startId 和 intent 的消息,消息的發(fā)送還是在主線程進(jìn)行的,接下來消息的接收、處理就是在子線程進(jìn)行的:

當(dāng)接收到消息時,通過 onHandleIntent() 方法在子線程處理 intent 對象, onHandleIntent() 方法執(zhí)行結(jié)束后,通過 stopSelf(msg.arg1) 等待所有消息處理完畢后終止服務(wù)。

為什么消息的處理是在子線程呢?這里涉及到 Handler 的內(nèi)部消息機(jī)制,簡單的說,因?yàn)?ServiceHandler 使用的 Looper 對象就是在 HandlerThread 這個子線程類里創(chuàng)建的,并通過 Looper.loop() 開啟消息循環(huán),不斷從消息隊(duì)列(單鏈表)中取出消息,并執(zhí)行,截取 loop() 的部分源碼:

dispatchMessage() 方法間接會調(diào)用 handleMessage() 方法,所以最終 onHandleIntent() 就在子線程中劃線執(zhí)行了,即 HandlerThread 的 run() 方法。

這就是 IntentService 實(shí)現(xiàn)的核心,通過 HandlerThread + Hanlder 把啟動 IntentService 的 Intent 從主線程切換到子線程,實(shí)現(xiàn)讓 Service 可以處理耗時任務(wù)的功能!

AsyncTask 是 Android 中輕量級的異步任務(wù)抽象類,它的內(nèi)部主要由線程池以及 Handler 實(shí)現(xiàn),在線程池中執(zhí)行耗時任務(wù)并把結(jié)果通過 Handler 機(jī)制中轉(zhuǎn)到主線程以實(shí)現(xiàn)UI操作。典型的用法如下:

從 Android3.0 開始,AsyncTask 默認(rèn)是串行執(zhí)行的:

如果需要并行執(zhí)行可以這么做:

AsyncTask 的源碼不多,還是比較容易理解的。根據(jù)上邊的用法,可以從 execute() 方法開始我們的分析:

看到 @MainThread 注解了嗎?所以 execute() 方法需要在主線程執(zhí)行哦!

進(jìn)而又調(diào)用了 executeOnExecutor() :

可以看到,當(dāng)任務(wù)正在執(zhí)行或者已經(jīng)完成,如果又被執(zhí)行會拋出異常!回調(diào)方法 onPreExecute() 最先被執(zhí)行了。

傳入的 sDefaultExecutor 參數(shù),是一個自定義的串行線程池對象,所有任務(wù)在該線程池中排隊(duì)執(zhí)行:

可以看到 SerialExecutor 線程池僅用于任務(wù)的排隊(duì), THREAD_POOL_EXECUTOR 線程池才是用于執(zhí)行真正的任務(wù),就是我們線程池部分講到的 ThreadPoolExecutor :

再回到 executeOnExecutor() 方法中,那么 exec.execute(mFuture) 就是觸發(fā)線程池開始執(zhí)行任務(wù)的操作了。

那 executeOnExecutor() 方法中的 mWorker 是什么? mFuture 是什么?答案在 AsyncTask 的構(gòu)造函數(shù)中:

原來 mWorker 是一個 Callable 對象, mFuture 是一個 FutureTask 對象,繼承了 Runnable 接口。所以 mWorker 的 call() 方法會在 mFuture 的 run() 方法中執(zhí)行,所以 mWorker 的 call() 方法在線程池得到執(zhí)行!

同時 doInBackground() 方法就在 call() 中方法,所以我們自定義的耗時任務(wù)邏輯得到執(zhí)行,不就是我們第二部分講的那一套嗎!

doInBackground() 的返回值會傳遞給 postResult() 方法:

就是通過 Handler 將最終的耗時任務(wù)結(jié)果從子線程發(fā)送到主線程,具體的過程是這樣的, getHandler() 得到的就是 AsyncTask 構(gòu)造函數(shù)中初始化的 mHandler , mHander 又是通過 getMainHandler() 賦值的:

可以在看到 sHandler 是一個 InternalHandler 類對象:

所以 getHandler() 就是在得到在主線程創(chuàng)建的 InternalHandler 對象,所以

就可以完成耗時任務(wù)結(jié)果從子線程到主線程的切換,進(jìn)而可以進(jìn)行相關(guān)UI操作了。

當(dāng)消息是 MESSAGE_POST_RESULT 時,代表任務(wù)執(zhí)行完成, finish() 方法被調(diào)用:

如果任務(wù)沒有被取消的話執(zhí)行 onPostExecute() ,否則執(zhí)行 onCancelled() 。

如果消息是 MESSAGE_POST_PROGRESS , onProgressUpdate() 方法被執(zhí)行,根據(jù)之前的用法可以 onProgressUpdate() 的執(zhí)行需要我們手動調(diào)用 publishProgress() 方法,就是通過 Handler 來發(fā)送進(jìn)度數(shù)據(jù):

進(jìn)行中的任務(wù)如何取消呢?AsyncTask 提供了一個 cancel(boolean mayInterruptIfRunning) ,參數(shù)代表是否中斷正在執(zhí)行的線程任務(wù),但是呢并不靠譜, cancel() 的方法注釋中有這么一段:

大致意思就是調(diào)用 cancel() 方法后, onCancelled(Object) 回調(diào)方法會在 doInBackground() 之后被執(zhí)行而 onPostExecute() 將不會被執(zhí)行,同時你應(yīng)該 doInBackground() 回調(diào)方法中通過 isCancelled() 來檢查任務(wù)是否已取消,進(jìn)而去終止任務(wù)的執(zhí)行!

所以只能自己動手了:

AsyncTask 整體的實(shí)現(xiàn)流程就這些了,源碼是最好的老師,自己跟著源碼走一遍有些問題可能就豁然開朗了!

文章題目:android子線程,Android子線程向主線程傳遞消息
標(biāo)題網(wǎng)址:http://chinadenli.net/article33/dsgceps.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供標(biāo)簽優(yōu)化企業(yè)建站網(wǎng)站改版靜態(tài)網(wǎng)站ChatGPT品牌網(wǎng)站制作

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都網(wǎng)頁設(shè)計(jì)公司