安卓异步任务 ---AsyncTask
为什么要异步任务:
1.Android单线程模型
2.耗时操作放在非主线程中执行
AsyncTask为何而生
1.子线程中更新UI
2.封装,简化异步操作
构造AsyncTask子类的参数
AsyncTask<Params, Progress, Result>是一个抽象类,通常用于被继承,继承AsyncTask需要指定如下三个泛型参数:
Params: 启动任务时输入的参数的类型,也就是execute()方法中,传入的参数
Progress: 后台任务执行中返回进度值的类型,进度更新时用到
Result 后台执行任务完成后返回结果的类型,如果后台是从网络上获取一张图片,那么就返回Bitmap类型
构建AsyncTask子类的回调方法
(插入知识:什么是回调函数)你到一个商店买东西,刚好你要的东西没有货,于是你在店员那里留下了你的电话,过了几天店里有货了,店员就打了你的电话,然后你接到电话后就到店里去取了货。在这个例子里,你的电话号码就叫回调函数,你把电话留给店员就叫登记回调函数,店里后来有货了叫做触发了回调关联的事件,店员给你打电话叫做调用回调函数,你到店里去取货叫做响应回调事件。(转自知乎 作者:常溪玲)
doInBackground 必须重写,异步执行后台线程将要完成的任务
onPreExecute 执行后台耗时操作前被调用,通常用户完成一些初始化操作
onPostExecute 当doInBackground()完成后,系统会自动调用,并将doInBackground方法返回的值传给该方法。
onProgressUpdate 在doInBackground()方法中调用publishProgress()方法更新任务的执行进度后,就会触发该方法。可以很清楚的了解耗时操作的完成进度
函数执行顺序演示代码:
MyAsynTask.java
package com.example.app; import android.os.AsyncTask; import android.util.Log; public class MyAsynTask extends AsyncTask<Void, Void, Void> { @Override protected Void doInBackground(Void... arg0) { // TODO Auto-generated method stub Log.d("sr", "doInBackground"); publishProgress(); return null; } @Override protected void onPostExecute(Void result) { // TODO Auto-generated method stub super.onPostExecute(result); Log.d("sr", "onPostExecute"); } @Override protected void onPreExecute() { // TODO Auto-generated method stub super.onPreExecute(); Log.d("sr", "onPreExecute"); } @Override protected void onProgressUpdate(Void... values) { // TODO Auto-generated method stub super.onProgressUpdate(values); Log.d("sr", "onProgressUpdate"); } }
package com.example.app; import android.os.Bundle; import android.app.Activity; import android.app.TaskStackBuilder; import android.view.Menu; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); MyAsynTask task = new MyAsynTask(); task.execute(); } }
输出结果:
下面是实例演示(加载一张网络图片):
异步线程:下载图像
UI线程:设置图像
还是在原来的工程中:
新建ImageTest.java类,继承Activity,创建OnCreate()方法。新建布局文件image.xml,里面有一个imageView显示图片 和一个进度条
image.xml
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:padding="16dp" android:orientation="vertical" > <ImageView android:id="@+id/image" android:layout_width="match_parent" android:layout_height="match_parent"/> <ProgressBar android:id="@+id/progressbar" android:visibility="gone" android:layout_centerInParent="true" android:layout_width="wrap_content" android:layout_height="wrap_content"/> </RelativeLayout>
package com.example.app; import android.app.Activity; import android.os.Bundle; import android.widget.ImageView; import android.widget.ProgressBar; public class ImageTest extends Activity { private ImageView mImageView; private ProgressBar mProgressBar; private static String URL = "http://image.zhaiba.com/2015/7/1/201507011732462088603781.jpg";//网路图片地址 @Override protected void onCreate(Bundle savedInstanceState) { // TODO Auto-generated method stub super.onCreate(savedInstanceState); setContentView(R.layout.image); mImageView = (ImageView) findViewById(R.id.image); mProgressBar = (ProgressBar)findViewById(R.id.progressbar); } }
接下来创建AsynTask异步处理
在ImageTest.java中创建内部类MyAsyncTask 继承AsyncTask
ImageTest.java
package com.example.app; import java.io.BufferedInputStream; import java.io.IOException; import java.io.InputStream; import java.net.MalformedURLException; import java.net.URL; import java.net.URLConnection; import android.app.Activity; 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; public class ImageTest extends Activity { private ImageView mImageView; private ProgressBar mProgressBar; private static String URL = "http://image.zhaiba.com/2015/7/1/201507011732462088603781.jpg";//网路图片地址 @Override protected void onCreate(Bundle savedInstanceState) { // TODO Auto-generated method stub super.onCreate(savedInstanceState); setContentView(R.layout.image); mImageView = (ImageView) findViewById(R.id.image); mProgressBar = (ProgressBar)findViewById(R.id.progressbar); //设置传递进去的参数 new MyAsyncTask().execute(URL);//可以传入一个,或多个参数 } class MyAsyncTask extends AsyncTask<String, Void, Bitmap> //第一个参数传入URL,所以是String,第二个参数不需要返回加载的进度,所以设置为Void,第三个值为返回值类型,这里是图片,所以设置为Bitmap { @Override protected void onPreExecute() { // TODO Auto-generated method stub super.onPreExecute(); //将隐藏的进度条显示出来,提示用户等待 mProgressBar.setVisibility(View.VISIBLE);//显示进度条 } @Override protected void onPostExecute(Bitmap result) { // TODO Auto-generated method stub super.onPostExecute(result); //操作UI 设置图像 mProgressBar.setVisibility(View.GONE); mImageView.setImageBitmap(result); } @Override protected Bitmap doInBackground(String... params) {//参数为不定长的数组,这是从execute()方法中传入的参数 // TODO Auto-generated method stub //获取传递进来的参数 String url = params[0];//上面execute方法中传入的参数第一个参数为URL Bitmap bitmap = null; URLConnection connection; InputStream is; try { connection = new URL(url).openConnection(); is = connection.getInputStream();//获取输入流 BufferedInputStream bis = new BufferedInputStream(is); //通过decodeStream方法解析输入流 bitmap = BitmapFactory.decodeStream(bis);//将输入流解析为图像 is.close(); bis.close(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } //将bitmap作为返回值返回后面调用的方法 return bitmap;//返回图像 } } }配置AndroidManifest增加网络权限,声明activity
在主页上添加一个按钮,点击,进入ImageTest那个activity
Activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" > <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:text="Image_Test" android:onClick="loadImage"/> </LinearLayout>
MainActivity.java
package com.example.app; import android.os.Bundle; import android.app.Activity; import android.app.TaskStackBuilder; import android.content.Intent; import android.view.Menu; import android.view.View; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); MyAsynTask task = new MyAsynTask(); task.execute(); } public void loadImage(View view) { startActivity(new Intent(this, ImageTest.class)); } }
下面的例子就是模拟进度条的更新。
进度条布局progressbar.xml
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:padding="16dp" android:orientation="vertical" > <ProgressBar style="?android:attr/progressBarStyleHorizontal" android:layout_width="match_parent" android:layout_height="wrap_content" android:id="@+id/pg" android:layout_centerVertical="true" android:layout_centerHorizontal="true"/> </RelativeLayout>
package com.example.app; import android.app.Activity; import android.os.AsyncTask; import android.os.Bundle; import android.widget.ProgressBar; public class ProgressBarTest extends Activity { private ProgressBar mProgressBar; private MyAsyncTask mTask; @Override protected void onCreate(Bundle savedInstanceState) { // TODO Auto-generated method stub super.onCreate(savedInstanceState); setContentView(R.layout.progressbar); mProgressBar = (ProgressBar)findViewById(R.id.pg); mTask = new MyAsyncTask(); mTask.execute(); } class MyAsyncTask extends AsyncTask<Void,Integer, Void> //第二个参数因为要实时的返回进度条的状态,所以要Integer参数 { @Override protected Void doInBackground(Void... arg0) { // TODO Auto-generated method stub //模拟进度更新 for(int i = 0; i < 100; i ++){ publishProgress(i); //去调用下面那个onProgressUpdate方法 try { Thread.sleep(300); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } return null; } @Override protected void onProgressUpdate(Integer... values) { // TODO Auto-generated method stub super.onProgressUpdate(values); //获取进度更新值 mProgressBar.setProgress(values[0]);//当前进度值 } } }
为主页添加一个测试进度条按钮,设置点击方法后,运行:
可以看到进度条在动态的更新
但是这样存在一个问题,返回后重新加载,再返回后重新加载,发现进度条不发生变化,因为每次返回加载都启动了一个异步任务,当前需要加载时有可能上次的任务还没有执行完,所以当前的进度条不更新。
把任务的生命周期设置为activity一样
在ProgressBarTest.java 中重写onPause()方法
@Override protected void onPause() { // TODO Auto-generated method stub super.onPause(); if(mTask != null && mTask.getStatus() == AsyncTask.Status.RUNNING){//任务不为空切正在运行 mTask.cancel(true); //cancel方法只是将对应的AsyncTask标记为cancel状态,并不是真正的取消线程的执行 } }
将当前任务设置为cancel状态,然后在doInBackground方法和onProgressUpdate方法中中判断一下
<span style="white-space:pre"> </span>protected Void doInBackground(Void... arg0) { // TODO Auto-generated method stub //模拟进度更新 for(int i = 0; i < 100; i ++){ if(isCancelled()) break; publishProgress(i); //去调用下面那个onProgressUpdate方法 try { Thread.sleep(300); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } return null; } @Override protected void onProgressUpdate(Integer... values) { // TODO Auto-generated method stub super.onProgressUpdate(values); if(isCancelled()) return ; //获取进度更新值 mProgressBar.setProgress(values[0]);//当前进度值 }
使用AsyncTask注意事项
必须在UI线程中创建AsyncTask的实例
必须在UI线程中调用AsyncTask的execute()方法
重写的四个方法时系统调用的,不能手动调用
每个AsyncTask只能被调用(execute())一次,多次调用将会引发异常(它的四个方法只有doInBackground是异步处理,不能直接更新UI,其他三个方法可以更新UI。
源代码下载:AsyncTask加载网路图片,模拟进度条的更新
附上网友写的一篇笔记:http://www.cnblogs.com/caobotao/p/5020857.html