您的位置:首页 > 移动开发 > Android开发

Android中AsyncTask(异步任务,实际就和handler的作用差不多)

2017-03-24 10:27 423 查看
Android API中有提到,AsyncTask非常适合短时间异步操作。如果要执行长时间操作,最好使用线程池Executor:

提到了两个原因:

AsyncTask的生命周期没有跟Activity的生命周期同步

容易内存泄露

AsyncTask和Activity的生命周期

      如果你在一个Activity中创建了一个AsyncTask,你旋转了屏幕,这个Activity将会被销毁并且会重新创建一个新的实例。但是AsyncTask没有销毁,它将会继续执行直到完成。当它执行完成后,它实际上是更新了上一个已经不存在的Activity,如果你原本想在onPostExecute()中更新UI的话,这时的AsyncTask将不会更新新的Activity,并且这个操作会引发一个异常:java.lang.IllegalArgumentException。

      如果你是想要在onPostExecute()中关闭一个dialog,则会发生:java.lang.IllegalArgumentException: View not attached to window manager(前提是你忘记在Activity的onStop()中忘记dismiss)。不仅如此,还会在show()的调用出抛出一个lead window的异常

内存泄露问题

      在Activity中作为内部类创建AsyncTask很方便。因为AsyncTask在执行完成或者执行中需要使用Activity中的view,因为内部类可以直接访问外部类的域(也就是变量)。然而这意味着内部类会持有外部类的一个引用。

      当长时间运行时,因为AsyncTask持有Activity的引用,所以即使当该Activity不再显示时Android也无法释放其占用的资源。

AsyncTask:(实际就和handler的作用差不多)

使用AsyncTask的具体步骤:注意3个泛型必须基本数据类型对象,也就是(Integer,String之类)

1.新建一个类,extends AsyncTask类<Params, Progress, Result>{}//三个泛型可用大写的Void替代(Void是小写void的对象类型)

2.复写其抽象方法,在异步任务(也就是doInBackground方法)执行之前,执行该方法,在主线程执行

protected void onPreExecute() {super.onPreExecute();}

3.复写抽象方法,该方法在后台执行(可以理解为子线程),可以做耗时的工作

protected Result doInBackground(Params... params){return null;}可以接收到doInBackground返回数据

4.复写其方法抽象方法,在异步任务(也就是doInBackground方法)执行之后,执行该方法,在主线程执行

protected void onPostExecute(Void result) {super.onPostExecute(result);}

5.开启一个异步任务,在主线程里新建继承了AsyncTask自定义的类,调用execute()方法.

        new myAsyncTasK().execute();//execute()方法一定要用,相当于开启线程的start,调了1个小时的bug.

 

关于AsyncTask参数的说明

1.类中的三个泛型:

    params:第一个泛型,就是5.execute方法的参数类型(其实就是他传的实际参数),也是3.doInBackground方法的参数类型

    提示:3.doInBackground(Params... params){内部使用时,params就是对应类型的数组,Prarms[0]就是第一个Params}

 

    Progress:第二个泛型,就是publishProgress(必须在3.doInBackground方法里调用)方法的参数类型,他负责发布进度,类似进度条,

    也是onProgressUpdate方法的参数类型,他得到的是来自publishProgress的实际参数,进度更新时调用,在主线程中执行,可更新UI

    提示:使用到第二个泛型,就要使用到publishProgress,并继续使用到onProgressUpdate(此方法要复写,具体格式如下:)

    protected void onProgressUpdate(Progress... values){super.onProgressUpdate(values);}//alues就是对应类型的数组

 

    Result:第三个泛型,既是3.doInBackground的返回值类型,并且它还是OnPostExecute的参数类型;

    应用场景:在子线程(doInBackground)获取信息,在OnpostExcute里去显示;

 

示例代码:

Android API中有提到,AsyncTask非常适合短时间异步操作。如果要执行长时间操作,最好使用线程池Executor:

提到了两个原因:

AsyncTask的生命周期没有跟Activity的生命周期同步

容易内存泄露

AsyncTask和Activity的生命周期

      如果你在一个Activity中创建了一个AsyncTask,你旋转了屏幕,这个Activity将会被销毁并且会重新创建一个新的实例。但是AsyncTask没有销毁,它将会继续执行直到完成。当它执行完成后,它实际上是更新了上一个已经不存在的Activity,如果你原本想在onPostExecute()中更新UI的话,这时的AsyncTask将不会更新新的Activity,并且这个操作会引发一个异常:java.lang.IllegalArgumentException。

      如果你是想要在onPostExecute()中关闭一个dialog,则会发生:java.lang.IllegalArgumentException: View not attached to window manager(前提是你忘记在Activity的onStop()中忘记dismiss)。不仅如此,还会在show()的调用出抛出一个lead window的异常

内存泄露问题

      在Activity中作为内部类创建AsyncTask很方便。因为AsyncTask在执行完成或者执行中需要使用Activity中的view,因为内部类可以直接访问外部类的域(也就是变量)。然而这意味着内部类会持有外部类的一个引用。

      当长时间运行时,因为AsyncTask持有Activity的引用,所以即使当该Activity不再显示时Android也无法释放其占用的资源。

AsyncTask:(实际就和handler的作用差不多)

使用AsyncTask的具体步骤:注意3个泛型必须基本数据类型对象,也就是(Integer,String之类)

1.新建一个类,extends AsyncTask类<Params, Progress, Result>{}//三个泛型可用大写的Void替代(Void是小写void的对象类型)

2.复写其抽象方法,在异步任务(也就是doInBackground方法)执行之前,执行该方法,在主线程执行

protected void onPreExecute() {super.onPreExecute();}

3.复写抽象方法,该方法在后台执行(可以理解为子线程),可以做耗时的工作

protected Result doInBackground(Params... params){return null;}可以接收到doInBackground返回数据

4.复写其方法抽象方法,在异步任务(也就是doInBackground方法)执行之后,执行该方法,在主线程执行

protected void onPostExecute(Void result) {super.onPostExecute(result);}

5.开启一个异步任务,在主线程里新建继承了AsyncTask自定义的类,调用execute()方法.

        new myAsyncTasK().execute();//execute()方法一定要用,相当于开启线程的start,调了1个小时的bug.

 

关于AsyncTask参数的说明

1.类中的三个泛型:

    params:第一个泛型,就是5.execute方法的参数类型(其实就是他传的实际参数),也是3.doInBackground方法的参数类型

    提示:3.doInBackground(Params... params){内部使用时,params就是对应类型的数组,Prarms[0]就是第一个Params}

 

    Progress:第二个泛型,就是publishProgress(必须在3.doInBackground方法里调用)方法的参数类型,他负责发布进度,类似进度条,

    也是onProgressUpdate方法的参数类型,他得到的是来自publishProgress的实际参数,进度更新时调用,在主线程中执行,可更新UI

    提示:使用到第二个泛型,就要使用到publishProgress,并继续使用到onProgressUpdate(此方法要复写,具体格式如下:)

    protected void onProgressUpdate(Progress... values){super.onProgressUpdate(values);}//alues就是对应类型的数组

 

    Result:第三个泛型,既是3.doInBackground的返回值类型,并且它还是OnPostExecute的参数类型;

    应用场景:在子线程(doInBackground)获取信息,在OnpostExcute里去显示;

 

示例代码:

public class MainActivity extends Activity {
private ProgressDialog pd;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 开启一个异步任务,在主线程里新建继承了AsyncTask自定义的类,调用execute()方法.
new myAsyncTasK().execute("www.dslkfja.com");
}

public class myAsyncTasK extends AsyncTask<String,Integer,ArrayList<String>>{
// 在异步任务执行之前,执行该方法,在主线程执行
protected void onPreExecute() {
System.out.println(Thread.currentThread().getName()+" "+"onPreExecute在工作");
super.onPreExecute();
pd = new ProgressDialog(MainActivity.this);
pd.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
pd.setMax(10);
pd.show();
}
// 该方法在后台执行(可以理解为子线程),可以做耗时的工作
protected ArrayList<String> doInBackground(String... params) {
System.out.println("网站是"+params[0]);
SystemClock.sleep(1000);
System.out.println(Thread.currentThread().getName()+" "+"doInBackground在后台忙碌");
for (int i = 0; i <10; i++) {
// 发布进度
publishProgress(i);
}
ArrayList<String> al=new ArrayList<String>();
al.add("aaa");
al.add("vvv");
al.add("ccc");
return al;
}
// 进度更新时调用,在主线程中执行,所以可以更新UI
protected void onProgressUpdate(Integer... values) {
super.onProgressUpdate(values);
pd.setProgress(values[0]);
System.out.println(Thread.currentThread().getName()+"我是onProgressUpdate"+values[0]);
}
// 在异步任务执行之后,执行该方法,在主线程执行
protected void onPostExecute(ArrayList<String> result) {
pd.dismiss();
super.onPostExecute(result);
System.out.println(result);
System.out.println(Thread.currentThread().getName()+" "+"onPostExecute在清理着垃圾");
Toast.makeText(getApplicationContext(), "进度条完成",0).show();
}
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: