Руководство Android AsyncTask
1. Android AsyncTask
Класс AsyncTask занесен в Android начиная с API Level 3, его цель задать стандартную цель для легкой работы с UI Thread.
Обычно AsyncTask используется для выполнения заданий в фоновом режиме в приложении и для обновления статуса данного задания на интерфейсе за все время выполнения задания.
В некоторых случаях вы используете UI для взаимодействия с AsyncTask, это создает некоторые риски. Точнее он может привести к утечке контекста (context), например пропущенные callback (обратные вызовы) или сбои при изменении конфигураций (Например пользователь поворачивает экран устройства). Он так же имеет противоречивое поведение на разных версиях платформы, сглатывает исключения от doInBackground и не предоставляет больше утилиты по сравнению с использованием напрямую Executor .
AsyncTask смоделирован как класс помогающий Thread и Handler. Его нужно использовать для короткиз действий (Примерно несколько секунд). Если вы хотите получить поток (thread) работающий долгое время, стоит использовать классы предоставленные java.util.concurrent, например Executor, ThreadPoolExecutor и FutureTask.
AsyncTask занесен в Android начиная с API Level 3, и отмечен как устаревший (deprecated) начиная с API Level 30 (Ạndroid 11).
Android AsyncTask Javadocs:Смотрите так же:
2. Пример AsyncTask
В данном примере мы будем использовать AsyncTask чтобы выполнить задание на фоновом режиме в приложении. И обновлять статус задания на интерфейсе за время его работы.
На Android Studio создайте новый project:
- File > New > New Project > Empty Activity
- Name: AsyncTaskExample
- Package name: org.o7planning.asynctaskexample
- Language: Java
Интерфейс приложения:
main_activity.xml
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<ProgressBar
android:id="@+id/progressBar"
style="?android:attr/progressBarStyleHorizontal"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="16dp"
android:layout_marginLeft="16dp"
android:layout_marginTop="32dp"
android:layout_marginEnd="16dp"
android:layout_marginRight="16dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/textView_info"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="16dp"
android:layout_marginLeft="16dp"
android:layout_marginTop="16dp"
android:layout_marginEnd="16dp"
android:layout_marginRight="16dp"
android:gravity="center"
android:text="Working info"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/progressBar" />
<Button
android:id="@+id/button_start"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="24dp"
android:text="Click to Start"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/textView_info" />
<Button
android:id="@+id/button_cancel"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="16dp"
android:text="Cancel"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/button_start" />
</androidx.constraintlayout.widget.ConstraintLayout>
MainActivity.java
package org.o7planning.asynctaskexample;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
private ProgressBar progressBar;
private TextView textViewInfo;
private Button buttonStart;
private Button buttonCancel;
private MyWorkTask myWorkTask;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
this.progressBar = (ProgressBar) this.findViewById(R.id.progressBar);
this.textViewInfo = (TextView) this.findViewById(R.id.textView_info);
this.buttonStart = (Button) this.findViewById(R.id.button_start);
this.buttonCancel = (Button) this.findViewById(R.id.button_cancel);
this.buttonStart.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startWork();
}
});
this.buttonCancel.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
requestCancel();
}
});
}
private void startWork() {
this.myWorkTask = new MyWorkTask(this.progressBar,
this.textViewInfo, this.buttonStart, this.buttonCancel);
ParamInfo param = new ParamInfo("Param 1", "Param 2");
this.myWorkTask.execute(param);
}
private void requestCancel() {
if(this.myWorkTask != null) {
this.myWorkTask.cancel(true);
}
}
}
MyWorkTask.java
package org.o7planning.asynctaskexample;
import android.os.AsyncTask;
import android.os.SystemClock;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;
import java.util.Date;
// <Params, Progress, Result>
public class MyWorkTask extends AsyncTask<ParamInfo, ProgressInfo, ResultInfo> {
private final ProgressBar progressBar;
private final TextView textViewInfo;
private final Button buttonStart;
private final Button buttonCancel;
private final int PROGRESS_MAX;
private int workCount = 0;
private long startTimeInMillis;
public MyWorkTask(ProgressBar progressBar, TextView textViewInfo,
Button buttonStart, Button buttonCancel) {
this.progressBar = progressBar;
this.textViewInfo = textViewInfo;
this.buttonStart = buttonStart;
this.buttonCancel = buttonCancel;
this.PROGRESS_MAX = this.progressBar.getMax();
}
@Override
protected void onPreExecute() {
this.progressBar.setVisibility(ProgressBar.VISIBLE);
this.textViewInfo.setText("Start...");
this.buttonStart.setEnabled(false);
this.buttonCancel.setEnabled(true);
this.startTimeInMillis = new Date().getTime();
}
@Override
protected ResultInfo doInBackground(ParamInfo... params) {
final int WORK_MAX = 30;
while (this.workCount < WORK_MAX) {
SystemClock.sleep(100); // 100 Milliseconds.
this.workCount++;
int progress = (this.workCount * PROGRESS_MAX) / WORK_MAX; // Progress value.
int percent = (progress * 100) / PROGRESS_MAX;
String info = "(" + percent +"%) - Working part " + this.workCount + " of " + WORK_MAX;
ProgressInfo progressInfo = new ProgressInfo(progress, info);
this.publishProgress(progressInfo); // Progress ...values
}
long finishTimeInMillis = new Date().getTime();
long workTimeInMillis = finishTimeInMillis - this.startTimeInMillis;
ResultInfo result = new ResultInfo(true, workTimeInMillis);
return result;
}
@Override
protected void onProgressUpdate(ProgressInfo... values) { // Progress ...values
ProgressInfo progressInfo= values[0];
int progress = progressInfo.getProgress();
this.progressBar.setProgress(progress);
this.textViewInfo.setText(progressInfo.getWorkingInfo());
}
@Override
protected void onPostExecute(ResultInfo resultInfo) {
super.onPostExecute(resultInfo);
this.buttonStart.setEnabled(true);
this.buttonCancel.setEnabled(false);
this.textViewInfo.setText(resultInfo.getMessage());
}
@Override
protected void onCancelled(ResultInfo resultInfo) {
super.onCancelled(resultInfo);
this.buttonStart.setEnabled(true);
this.buttonCancel.setEnabled(false);
this.textViewInfo.setText(resultInfo.getMessage());
}
}
ParamInfo.java
package org.o7planning.asynctaskexample;
public class ParamInfo {
private String param1;
private String param2;
public ParamInfo(String param1, String param2) {
this.param1 = param1;
this.param2 = param2;
}
public String getParam1() {
return param1;
}
public String getParam2() {
return param2;
}
}
ProgressInfo.java
package org.o7planning.asynctaskexample;
public class ProgressInfo {
private int progress;
private String workingInfo;
public ProgressInfo(int progress, String workingInfo) {
this.progress = progress;
this.workingInfo = workingInfo;
}
public int getProgress() {
return progress;
}
public String getWorkingInfo() {
return workingInfo;
}
}
ResultInfo.java
package org.o7planning.asynctaskexample;
public class ResultInfo {
private boolean completed;
private long workTimeInMillis;
public ResultInfo(boolean completed, long workTimeInMillis) {
this.completed = completed;
this.workTimeInMillis = workTimeInMillis;
}
public boolean isCompleted() {
return completed;
}
public long getWorkTimeInMillis() {
return workTimeInMillis;
}
public String getMessage() {
if(this.completed) {
return "Complete in " + this.workTimeInMillis +" milliseconds";
}
return "Failed or cancelled";
}
}
Pуководства Android
- Настроить Android Emulator в Android Studio
- Руководство Android ToggleButton
- Создать простой File Finder Dialog в Android
- Руководство Android TimePickerDialog
- Руководство Android DatePickerDialog
- Что мне нужно для начала работы с Android?
- Установите Android Studio в Windows
- Установите Intel® HAXM для Android Studio
- Руководство Android AsyncTask
- Руководство Android AsyncTaskLoader
- Руководство Android для начинающих - основные примеры
- Как узнать номер телефона Android Emulator и изменить его?
- Руководство Android TextInputLayout
- Руководство Android CardView
- Руководство Android ViewPager2
- Получить номер телефона в Android с помощью TelephonyManager
- Руководство Android Phone Call
- Руководство Android Wifi Scanning
- Руководство Android 2D Game для начинающих
- Руководство Android DialogFragment
- Руководство Android CharacterPickerDialog
- Руководство Android для начинающих - Hello Android
- Использование Android Device File Explorer
- Включить USB Debugging на устройстве Android
- Руководство Android UI Layouts
- Руководство Android SMS
- Руководство Android SQLite Database
- Руководство Google Maps Android API
- Руководство Текст в речь на Android
- Руководство Android Space
- Руководство Android Toast
- Создание пользовательских Android Toast
- Руководство Android SnackBar
- Руководство Android TextView
- Руководство Android TextClock
- Руководство Android EditText
- Руководство Android TextWatcher
- Форматирование номера кредитной карты с помощью Android TextWatcher
- Руководство Android Clipboard
- Создать простой File Chooser в Android
- Руководство Android AutoCompleteTextView и MultiAutoCompleteTextView
- Руководство Android ImageView
- Руководство Android ImageSwitcher
- Руководство Android ScrollView и HorizontalScrollView
- Руководство Android WebView
- Руководство Android SeekBar
- Руководство Android Dialog
- Руководство Android AlertDialog
- Руководство Android RatingBar
- Руководство Android ProgressBar
- Руководство Android Spinner
- Руководство Android Button
- Руководство Android Switch
- Руководство Android ImageButton
- Руководство Android FloatingActionButton
- Руководство Android CheckBox
- Руководство Android RadioGroup и RadioButton
- Руководство Android Chip и ChipGroup
- Использование Image assets и Icon assets Android Studio
- Настройка SD Card для Android Emulator
- Пример ChipGroup и Chip Entry
- Как добавить внешние библиотеки в Android Project в Android Studio?
- Как отключить разрешения, уже предоставленные приложению Android?
- Как удалить приложения из Android Emulator?
- Руководство Android LinearLayout
- Руководство Android TableLayout
- Руководство Android FrameLayout
- Руководство Android QuickContactBadge
- Руководство Android StackView
- Руководство Android Camera
- Руководство Android MediaPlayer
- Руководство Android VideoView
- Воспроизведение звуковых эффектов в Android с помощью SoundPool
- Руководство Android Networking
- Руководство Android JSON Parser
- Руководство Android SharedPreferences
- Руководство Android Internal Storage
- Руководство Android External Storage
- Руководство Android Intents
- Пример явного Android Intent, вызов другого Intent
- Пример неявного Android Intent, откройте URL, отправьте email
- Руководство Android Services
- Использовать оповещения в Android - Android Notification
- Руководство Android DatePicker
- Руководство Android TimePicker
- Руководство Android Chronometer
- Руководство Android OptionMenu
- Руководство Android ContextMenu
- Руководство Android PopupMenu
- Руководство Android Fragment
- Руководство Android ListView
- Android ListView с Checkbox с помощью ArrayAdapter
- Руководство Android GridView
Show More