Android 的 UI线程主要负责处理用户的按键事件、触屏事件等,因此其他阻塞UI线程的操作不应该在主线程中进行。
为了避免 UI线程失去响应的间题,Android 程序采用将耗时操作放在新线程中完成的方式,但是新线程间能需要动态更新UI组件,比如获取网络资源操作放在新线程中完成。由于新线程不允许直接更新UI组件,为了解决这个间题, Android 提供了以下儿种方式:
使用 Hanlder 实现线程之间的通信。
View . post ( Runnable )。
Activity . runOnUiThread ( Runnable )。
上述方式使用起来有点复杂,采用异步任务( AsyncTask )则可以进一步简化操作。相对来说,异步任务 AsyneTask 更轻量级一些,适用于简单的异步任务,例如获取网络数据View 、动态更改UI界面等。
AsyncTask<Params,Progress,Result>是一个抽象类,通常用于被继承,继承时需要指定如下3个泛型参数:
Params -启动任务执行的输人参数的类型。
Progress- 后台任务完成的进度值的类型。
Result -后台任务执行完成以后返回结果的类型。
AsyncTask 的步骤如下:
(1)创建 AsyncTask 的子类,并指定参数类型。如果某个参数不需要,则指定为 Void 。
(2)实现 AsyncTask 的方法,姆 dolnBackground ( Params .….):后台线程将要完成的功能,一般用于获取网络资源等耗时性的操作;第二个方法是 onPostExecute ( Result result ):在 dolnBackground()方法执行完以后,系统会自动调用 onPostExecute ()方法,并接收其返回值。这里一般负责更新 UI 线程等操作。
(3)调用 AsyncTask 子类的实例的 execute ( Params . params )方法,执行耗时操作。
接下来通过一个从网络下载图片的具体例子来讲解如何使用 AsyncTask类。
首先是界面布局,这里采用相对布局的方式,只需要一个 ImageView 显 示和一个 progressBar 以查看是否下载完成。
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/RelativeLayout1"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity" >
<ImageView
android:id="@+id/imageView"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_alignParentLeft="true"
android:layout_alignParentTop="true" />
<ProgressBar
android:id="@+id/progressBar"
android:visibility="gone"
style="?android:attr/progressBarStyleLarge"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_centerVertical="true" />
</RelativeLayout>
android:visibility="gone"采用了隐藏属性。
package com.example.ansynctask;
import androidx.appcompat.app.AppCompatActivity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.os.Bundle;
import android.view.View;
import android.widget.ImageView;
import android.widget.ProgressBar;
import java.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
public class MainActivity extends AppCompatActivity {
private ImageView mImageView=null;
private ProgressBar mProgressBar=null;
private String URLs="http://wallcoo.com/paint/fine-art-paintings/wallpapers/1280x1024/art-painting-vangogh_starry_wallcoo.com.jpg";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.layout);
//实例化异步任务
this.mImageView=(ImageView)findViewById(R.id.imageView);
this.mProgressBar=(ProgressBar)findViewById(R.id.progressBar);
ImageDownloadTask task=new ImageDownloadTask();
//执行异步任务
task.execute(URLs);
}
class ImageDownloadTask extends AsyncTask<String,Void, Bitmap> {
@Override
protected Bitmap doInBackground(String... params) {
Bitmap bitmap = null; //待返回的结果
String url = params[0]; //获取URL
URLConnection connection; //网络连接对象
InputStream is; //数据输入流
try {
connection = new URL(url).openConnection();
is = connection.getInputStream();
BufferedInputStream buf = new BufferedInputStream(is);
//解析输入流
bitmap = BitmapFactory.decodeStream(buf);
is.close();
buf.close();
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
//返回给后面调用的方法
return bitmap;
}
@Override
protected void onPreExecute()
{
//显示等待圆环
mProgressBar.setVisibility(View.VISIBLE);
}
@Override
protected void onPostExecute(Bitmap result) {
//下载完毕,隐藏等待圆环
mProgressBar.setVisibility(View.GONE);
mImageView.setImageBitmap(result);
}
}
}
最后要在 AndroidManifest 中加上网络访问权限,代码如下所示:
<uses-permission android:name="android.permission.INTERNET"></uses-permission>
注:API在29以上,AsyncTask标记为Deprecated。