AsyncTask实现异步任务
使用Handler类来在子线程中更新UI线程总会启动一些莫名的子线程,太多的子线程给系统带来巨大的负担。
Android提供了一个工具类AsyncTask,来实现异步执行任务。
AsyncTask是抽象类,具有三种泛型:Params
,Progress
和Result
Params
:表示启动任务执行的输入参数,比如HTTP请求的URL
Progress
:表示后台任务执行的百分比
Result
:表示后台执行任务返回的最终的结果,比如String,Integer等
实现
通过继承一个AsyncTask类定义一个异步任务类
Android提供一个程序员编写后台操作更为容易和透明AsyncTask,使得后台线程能够在UI主线程外进行处理
使用AsyncTask,不需要自己来写后台线程,无需终结后台线程,只需要创建AsyncTask类,并实现其中的抽象方法以及重写某些方法
。
实现步骤:
- 使用execute方法触发异步任务的执行
- 使用onPreExecute()表示执行预处理:绘制一个进度条控件
- 使用doInBackground()用于执行较为费时的操作:计算进度 - - 这个方法是AsyncTask的关键,必须覆盖重写
- 使用onProgressUpdate()对进度条控件根据进度值做出具体的响应
- 使用onPostExecute()可以对后台任务的结果做出处理
MainActivity.java 代码
package com.example.asynctaskdemo;
import androidx.appcompat.app.AppCompatActivity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
Button button;
ProgressBar progressBar;
TextView textView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
button=findViewById(R.id.button);
progressBar=findViewById(R.id.progressBar);
textView=findViewById(R.id.textView);
button.setOnClickListener(this);
}
@Override
public void onClick(View v) {
TimerTickLoad timerTickLoad=new TimerTickLoad();
timerTickLoad.execute(1000);
}
class TimerTickLoad extends AsyncTask<Integer,Integer,String>{
@Override
protected void onPreExecute() {
super.onPreExecute();
textView.setText("onPreExecute");
Log.d("hello","onPreExecute");
progressBar.setVisibility(ProgressBar.VISIBLE);//初始化的时候显示进度条
}
@Override
protected void onPostExecute(String s) {
super.onPostExecute(s);
Log.d("hello","onPostExecute");
textView.setText(s);
}
@Override
protected void onProgressUpdate(Integer... values) {
super.onProgressUpdate(values);
textView.setText("onProgressUpdate");
progressBar.setProgress(values[0]);
Log.d("hello","onProgressUpdate");
}
@Override
protected String doInBackground(Integer... integers) {
Log.d("hello","doInBackground");
//不能进行UI操作
for(int i=0;i<=10;i++){
publishProgress(i*10);//调用刷新进度条
try {
Thread.sleep(integers[0]);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
return "执行完毕";
}
}
}
acitivity_main.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">
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<Button
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="28dp"
android:layout_marginLeft="28dp"
android:text="Press it"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="0.074" />
<ProgressBar
android:id="@+id/progressBar"
style="?android:attr/progressBarStyleHorizontal"
android:layout_width="0dp"
android:layout_height="34dp"
android:layout_marginTop="24dp"
android:visibility="invisible"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/button" />
</androidx.constraintlayout.widget.ConstraintLayout>