1印蓖、AsyncTask簡介
AsyncTask允許你能夠很容易并且正確地訪問UI線程筐摘。它能夠讓你輕松地執(zhí)行后臺操作并且把結(jié)果返回給UI線程,它的目標(biāo)是代替Thread和Handler來執(zhí)行輕量化地跨線程操作襟沮。
2诬像、使用場景
Android系統(tǒng)禁止任何主線程的網(wǎng)絡(luò)連接行為,即時強(qiáng)行為之畏梆,Android也會拋出NetworkMainThreadException異常您宪;同時,如果我們在主線程中執(zhí)行長時間的耗時任務(wù)奠涌,在等待響應(yīng)期間宪巨,用戶界面毫無反應(yīng),這樣很可能導(dǎo)致應(yīng)用無響應(yīng)(Application Not Responding铣猩, ANR)現(xiàn)象發(fā)生揖铜。因此,我們可以將網(wǎng)絡(luò)連接等耗時任務(wù)放到后臺線程中去執(zhí)行达皿,執(zhí)行完之后再回到主線程中將結(jié)果展示出來天吓。怎樣使用后臺線程最容易贿肩?答案是使用AsyncTask。與Thread和Handler相比龄寞,AsyncTask是如此的簡單
3.AsyncTask使用簡介
1)如果獲取AsyncTask
AsyncTask是一個抽象類汰规,我們使用時需要繼承它,然后實現(xiàn)它的抽象方法物邑。AsyncTask強(qiáng)制要求我們實現(xiàn)的只有一個方法溜哮,即doInBackground(),通過名稱就可以知道色解,可以把我們想要在后臺執(zhí)行的操作放到該方法中茂嗓。一個最簡單的AsyncTask實現(xiàn)如下
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
new AsyncTaskImpl().execute();
}
private class AsyncTaskImpl extends AsyncTask<Void, Void, Void>{
@Override
protected Void doInBackground(Void... params) {
return null;
}
}
}
2)參數(shù)解析
從上面的創(chuàng)建Async的過程中,我們看到AsyncTask實現(xiàn)時需要傳入三個類型參數(shù)科阎,下面我們一一解析一下
a述吸、第一個參數(shù)
有時候我們執(zhí)行的后臺操作可能需要我們傳入一些參數(shù)才能正常運(yùn)行,在這種情況下锣笨,我們就可以使用第一個參數(shù)為后臺操作指導(dǎo)參數(shù)的類型蝌矛,我們暫時稱它為執(zhí)行參數(shù)。既然是傳遞給后臺操作的參數(shù)類型错英,可想而知入撒,doInBackground()方法的形參也會作出相應(yīng)的變化,此時doInBackground()方法的形參就變成我們制定類型的變長參數(shù)了椭岩。一個帶執(zhí)行參數(shù)的AsyncTask實現(xiàn)如下:
public class MainActivity extends AppCompatActivity {
private static final String TAG = "MainActivity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
new AsyncTaskImpl().execute("AAAA","BBBB","CCCC");
}
private class AsyncTaskImpl extends AsyncTask<String, Void, Void>{
@Override
protected Void doInBackground(String... params) {
for (String s : params){
Log.d(TAG, s);
}
return null;
}
}
}
同時我們打印了在doInBackground()方法獲取到的參數(shù)茅逮,控制臺輸出如下:
06-26 16:19:11.432 12416-12448/com.android.yanghuaan.asynctaskdemo D/MainActivity: AAAA
06-26 16:19:11.432 12416-12448/com.android.yanghuaan.asynctaskdemo D/MainActivity: BBBB
06-26 16:19:11.432 12416-12448/com.android.yanghuaan.asynctaskdemo D/MainActivity: CCCC
b、第三個參數(shù)
??為什么直接跳過第二個參數(shù)介紹第三個參數(shù)簿煌?以為我覺得第三個參數(shù)對我們來說更重要或者說更實用氮唯,第二個參數(shù)后面再介紹。我們執(zhí)行后臺操作姨伟,很多情況下是希望從后臺操作中獲取我們想要的數(shù)據(jù)惩琉,這正是第三個參數(shù)要干的事情:指定返回參數(shù)類型《峄模看到這里瞒渠,不盡有人想問,我們從哪里獲得返回的數(shù)據(jù)呢技扼?其實伍玖,AsyncTask還有一個方法:onPostExecute(),這個方法只提供了空的實現(xiàn)剿吻,我們需要從后臺操作中獲得返回的數(shù)據(jù)窍箍,就必須覆蓋這個方法。值得注意的是,這個方法的是在主線程中執(zhí)行的椰棘,因此我們可以在這個方法中做主線程可以才能做的事情纺棺,比如修改UI元素等。一個簡單的示例如下:
public class MainActivity extends AppCompatActivity {
private TextView mTextView;
private static final String TAG = "MainActivity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTextView = (TextView) findViewById(R.id.text_view);
mTextView.setText("helloWorld");
new AsyncTaskImpl().execute("AAAA","BBBB","CCCC");
}
private class AsyncTaskImpl extends AsyncTask<String, Void, List<String>>{
@Override
protected List<String> doInBackground(String... params) {
List<String> result = new ArrayList<>();
for (int i = params.length - 1; i >= 0; i --){
result.add(params[i]);
}
return result;
}
@Override
protected void onPostExecute(List<String> strings) {
mTextView.setText("in onPostExecute method");
for (String s : strings){
Log.d(TAG, s);
}
}
}
}
控制臺輸出如下:
06-26 16:44:25.179 8647-8647/? D/MainActivity: CCCC
06-26 16:44:25.180 8647-8647/? D/MainActivity: BBBB
06-26 16:44:25.180 8647-8647/? D/MainActivity: AAAA
??同時可以看到屏幕上的顯示的內(nèi)容由默認(rèn)的helloWorld變?yōu)閕n onPostExecute method邪狞。
c祷蝌、第二個參數(shù)
如果說我們想要知道后臺操作進(jìn)行的進(jìn)程如何,該怎么辦呢帆卓?第二個參數(shù)就可以用來指定發(fā)送進(jìn)度更新需要的數(shù)據(jù)類型巨朦。這個參數(shù)與Async的另一個方法onProgressUpdate()有關(guān),即該參數(shù)用來指定此方法的形參剑令,并且這個方法也是在主線程中執(zhí)行的糊啡,可以用來修改UI元素,最常見的就是設(shè)置進(jìn)度條進(jìn)度吁津。
??值得注意的是悔橄,進(jìn)度更新通常放生在執(zhí)行后臺線程中。問題是后臺線程無法完成必要的UI更新腺毫。因此AsyncTask提供了publishProgress()和onProgressUpdate()方法。其工作方式是:從doInBackground()方法中調(diào)用publishProgress()方法挣柬,這樣onProgressUpdate()方法就能在主線程中更新UI了潮酒,即傳入publishProgress()方法的參數(shù)最終傳遞給了onProgressUpdate()方法,而第二個參數(shù)就是用來指定此參數(shù)的類型邪蛔。一個完整的示例如下:
public class MainActivity extends AppCompatActivity {
private TextView mTextView;
private ProgressBar mProgressBar;
private static final String TAG = "MainActivity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTextView = (TextView) findViewById(R.id.text_view);
mTextView.setText("helloWorld");
mProgressBar = (ProgressBar) findViewById(R.id.progress_bar);
mProgressBar.setProgress(0);
new AsyncTaskImpl().execute("AAAA","BBBB","CCCC");
}
private class AsyncTaskImpl extends AsyncTask<String, Integer, List<String>>{
@Override
protected List<String> doInBackground(String... params) {
List<String> result = new ArrayList<>();
for (int i = params.length - 1; i >= 0; i --){
result.add(params[i]);
}
for (int i = 1; i <= 100; i ++){
try{
Thread.sleep(100);
}catch (InterruptedException ie){
ie.printStackTrace();
}
publishProgress(i);
}
return result;
}
@Override
protected void onPostExecute(List<String> strings) {
for (String s : strings){
Log.d(TAG, s);
}
}
@Override
protected void onProgressUpdate(Integer... values) {
int progress = values[0];
mProgressBar.setProgress(progress);
if (progress == 100){
mTextView.setText("Complete.");
}
}
}
}
可以看到屏幕上的進(jìn)度條逐漸增加并且最終默認(rèn)的helloWorld變?yōu)镃omplete急黎。