AsyncTask 從概念到實現
概念
AsyncTask 是一個允許在後臺執行操作的類,其結果在 UI 執行緒上釋出。主要目的是通過消除處理程式和操作執行緒所需的所有東西來消除啟動/執行執行緒的所有樣板程式碼。此外,AsyncTask 的目的是在後臺執行緒上進行短時操作(最多幾秒),而不是長時間操作。因此,重要的是 AsyncTask 不要與通用執行緒框架混淆。如果需要進行長時間操作,則建議使用併發包。
一般考慮
AsyncTask 由三種泛型型別定義:引數,進度和結果。從執行的那一刻起,它經歷了 4 個步驟(方法)。首先是 onPreExecute ,有人可以定義一個載入對話方塊,或者一些可以通知使用者執行即將開始的 UI 訊息。接下來, doInBackground 是在與 Ui 執行緒不同的執行緒上非同步執行的方法。第三種方法是 onProgressUpdate ,它也可以在 UI 執行緒上執行,該執行緒可以通知使用者狀態。呼叫它的最後一個方法是 ***onPostExecute,***它主要用於釋出結果。
下面是一個如何使用 AsyncTask 返回字串的示例。
例 1
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button button = (FloatingActionButton) findViewById(R.id.btn);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
executeAsyncTaskOperation();
}
});
}
private void executeAsyncTaskOperation() {
new CustomAsyncTask(this).execute();
}
private static class CustomAsyncTask extends AsyncTask<Void, Void, String> {
private Context context;
private ProgressDialog progressDialog;
public CustomAsyncTask(Context context) {
this.context = context;
}
@Override
protected void onPreExecute() {
progressDialog = ProgressDialog.show(context, "Please wait...", "Loading data from web");
}
@Override
protected String doInBackground(Void... params) {
String object = null;
try {
Log.d(CustomAsyncTask.class.getCanonicalName(), "doInBackground");
Thread.sleep(500);
//bject = "new object";
} catch (Exception exc) {
Log.e(CustomAsyncTask.class.getCanonicalName(), "exception");
object = null;
}
return object;
}
@Override
protected void onPostExecute(String s) {
if (progressDialog != null && progressDialog.isShowing()) {
progressDialog.dismiss();
}
if (s != null) {
Toast.makeText(context, "finished successfully!", Toast.LENGTH_LONG).show();
} else {
Toast.makeText(context, "finished unsuccessfully!", Toast.LENGTH_LONG).show();
}
}
}
}
例 2
這裡,AsyncTask 有點不同,execute 方法接收要在後臺分析的資料列表。返回結果取決於此檢查。
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
executeAsyncTaskOperation();
}
});
}
private void executeAsyncTaskOperation() {
Boolean[] bools = new Boolean[10];
for (int k = 0; k < 10; k++) {
if (k % 2 == 0) {
bools[k] = true;
} else {
bools[k] = false;
}
}
new CustomAsyncTask(this).execute(bools);
}
private static class CustomAsyncTask extends AsyncTask<Boolean, Void, Integer> {
private Context context;
private ProgressDialog progressDialog;
public CustomAsyncTask(Context context) {
this.context = context;
}
@Override
protected void onPreExecute() {
progressDialog = ProgressDialog.show(context, "Please wait...", "Loading data from web");
}
@Override
protected Integer doInBackground(Boolean... params) {
int count = 0;
try {
Thread.sleep(1000);
Log.d(CustomAsyncTask.class.getCanonicalName(), "doInBackground");
for (Boolean param : params) {
if (param) {
count++;
}
}
} catch (Exception exc) {
Log.e(CustomAsyncTask.class.getCanonicalName(), "exception");
count = 0;
}
return count;
}
@Override
protected void onPostExecute(Integer s) {
if (progressDialog != null && progressDialog.isShowing()) {
progressDialog.dismiss();
}
if (s != null && s > 0) {
Toast.makeText(context, "finished loading: " + s + " tasks", Toast.LENGTH_LONG).show();
} else {
Toast.makeText(context, "finished unsuccessfully!", Toast.LENGTH_LONG).show();
}
}
}
}