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

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();
        }
    }
}





发布了94 篇原创文章 · 获赞 37 · 访问量 25万+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章