AsyncTask,顧名思義,異步任務(wù)。說到異步,最簡(jiǎn)單的理解就是不同步。再?gòu)?fù)雜一點(diǎn)理解,就得舉例子了。
假設(shè)我要去火車站買票,剛到火車站我突然發(fā)現(xiàn)我忘了帶身份證。怎么辦?怎么辦!
想辦法,想辦法!我想我應(yīng)該找個(gè)在學(xué)校的同學(xué)幫我送過來,因?yàn)槲也荒茏约夯厝ツ冒。€要排隊(duì)呢,走不開。嗯,要找人送過來。但是問題來了,我找人送身份證了,我去排隊(duì)了,如果排到第一位了身份證還沒到怎么辦?叮,腦袋上面突然亮了一個(gè)小燈泡,機(jī)智的我在排隊(duì)前想到了兩種方案:
第一種方案,讓售票員等著我,我后面隊(duì)伍里買票的人也等著我,我一直在窗口第一位置等著同學(xué)來送身份證,直到,我的身份證被送來,然后順利買票。
另一種方案呢,就是我跟售票員說一下,讓我在一邊等著送身份證,后面的人繼續(xù)買票,等我的身份證送來的時(shí)候我通知下售票員,就可以盡快排到隊(duì)伍第一位(不一定是立即排到第一位,因?yàn)槿f一有人正在買票,我不能過去打斷他)然后買票。
所以呢,選第一種還是第二種?我肯定選第二種,因?yàn)檫x第一種肯定會(huì)被后面排隊(duì)的人罵死,而且還有可能被售票員罵,搞不好還會(huì)挨揍,畢竟因?yàn)槲乙粋€(gè)人,浪費(fèi)了這么多人的時(shí)間,也拖慢了售票員的工作效率。
好了,例子就說到這里。在例子里,第二種方法就是異步的。異步往往和多線程有關(guān),而且異步任務(wù)也大多是交由一個(gè)單獨(dú)的線程完成,然后返回結(jié)果給主線程。這里售票員相當(dāng)于cpu,而排隊(duì)買票的人相當(dāng)于等待被執(zhí)行的任務(wù),而我是個(gè)被標(biāo)記為異步的任務(wù)(因?yàn)槲抑牢規(guī)矸葑C,不能立即買到票,所以排隊(duì)前就想好了第二種方案),當(dāng)cpu執(zhí)行到我這個(gè)任務(wù)的時(shí)候,發(fā)現(xiàn)我這個(gè)任務(wù)可執(zhí)行的條件(身份證)不具備,所以由我發(fā)起了一個(gè)異步任務(wù)(同學(xué)送票),去獲取可執(zhí)行的條件,之后立即把位置讓出來,讓其他排隊(duì)的任務(wù)繼續(xù)執(zhí)行。直到我的身份證拿來,然后立馬通知cpu準(zhǔn)備接待我。
嗯,差不多就是這樣了。開始說正文,android里面的AsyncTask。先上一段官網(wǎng)的引用:
AsyncTask enables proper and easy use of the UI thread. This class allows to perform background operations and publish results on the UI thread without having to manipulate threads and/or handlers.
是英語(yǔ),雖然我能懂大概是什么意思,但是還是不翻譯了,怕誤人子弟,等我英語(yǔ)學(xué)的再好些再來翻譯吧。不過還是要解釋下大概的意思,就是說AsyncTask可以在UI線程上做一些后臺(tái)操作,也能返回操作結(jié)果到UI線程上。我們知道UI線程是不能做一些耗時(shí)的操作的,但是有了AsyncTask,我們可以這樣做了。但是,
AsyncTask is designed to be a helper class around Thread and Handler and does not constitute a generic threading framework. AsyncTasks should ideally be used for short operations (a few seconds at the most.) If you need to keep threads running for long periods of time, it is highly recommended you use the various APIs provided by the java.util.concurrent package such as Executor, ThreadPoolExecutor and FutureTask.
對(duì)于耗時(shí)比較久的任務(wù),還是不建議放在AsyncTask中執(zhí)行。AysncTask被設(shè)計(jì)成Thread和Handler的輔助類,并不能執(zhí)行過于復(fù)雜和耗時(shí)的任務(wù),這種任務(wù)應(yīng)該用其他方法這里就不說了。AsyncTask最好用于耗時(shí)最多只有幾秒鐘的操作,比如向網(wǎng)絡(luò)請(qǐng)求個(gè)xml或是json之類的網(wǎng)絡(luò)操作,或是用在程序的初始化界面等等。
下面說下AsyncTask的使用。其實(shí)很簡(jiǎn)單。
首先你需要定義一個(gè)AsyncTask的子類,并且必須重寫父類的doInBackground(Params...)方法。另外還有onPostExecute(Result)方法也可重寫,這個(gè)方法在doInBackground之后被自動(dòng)調(diào)用,所以你可以在這里寫一些任務(wù)完成的通知代碼。
先給一個(gè)官方的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
private class DownloadFilesTask extends AsyncTask<URL, Integer, Long> { protected Long doInBackground(URL... urls) { int count = urls.length; long totalSize = 0 ; for ( int i = 0 ; i < count; i++) { totalSize += Downloader.downloadFile(urls[i]); publishProgress(( int ) ((i / ( float ) count) * 100 )); // Escape early if cancel() is called if (isCancelled()) break ; } return totalSize; } protected void onProgressUpdate(Integer... progress) { setProgressPercent(progress[ 0 ]); } protected void onPostExecute(Long result) { showDialog( "Downloaded " + result + " bytes" ); } } |
執(zhí)行AsyncTask的時(shí)候,必須在UI線程中執(zhí)行,如下語(yǔ)句。
new DownloadFilesTask().execute(url1, url2, url3);
可以看到,在繼承AsyncTask的時(shí)候,有幾個(gè)泛型類型,如AsyncTask<URL, Integer, Long>,簡(jiǎn)單解釋下。
第一個(gè)可以指定輸入?yún)?shù)的類型,就是new DownloadFilesTask().execute()的參數(shù)(最后傳到了doInBackground),這里的參數(shù)可以不只一個(gè),因?yàn)樽詈蟮椒椒ɡ锩妫盏降氖莻€(gè)數(shù)組。
第二個(gè)可指定發(fā)送進(jìn)度更新需要的類型,一般都是整型,用在publishProgress(用來在后臺(tái)進(jìn)程中發(fā)送進(jìn)度的方法,直接使用的,不用定義)和onProgressUpdate兩個(gè)方法中。
第三個(gè)是AsyncTask返回結(jié)果的數(shù)據(jù)類型,它設(shè)置了doInBackground的返回類型,以及onPostExecute的輸入?yún)?shù)類型
當(dāng)然,如果你什么都不需要,可以都使用Void。
private class MyTask extends AsyncTask<Void, Void, Void> { ... }
另外,還有一個(gè)可以重寫的方法,是onPreExecute(),它在doInBackground之前被調(diào)用,所以如果需要的話,你可以重寫它然后做一些實(shí)例化進(jìn)度條啊之類的工作。
最后,總結(jié)一下:
使用AsyncTask,你要做的是,繼承父類,然后重寫doInBackground(Params...),在里面實(shí)現(xiàn)后臺(tái)操作,如果有返回結(jié)果的話,重寫onPostExecute(Result)然后處理后臺(tái)程序的結(jié)果。
如果需要更新進(jìn)度的話,在onPreExecute()里實(shí)例化進(jìn)度條(也可以不在這),之后在doInBackground(Params...)里面用publishProgress()發(fā)布進(jìn)度值,然后重寫onProgressUpdate(Progress...)接收onPreExecute()發(fā)布的結(jié)果,并添加更新進(jìn)度條的代碼。
以上就是關(guān)于Android中異步任務(wù)的內(nèi)容了,希望大家能夠喜歡。