现在的位置: 首页 > 移动开发 > 正文

Android 异步操作AsyncTask

2019年09月10日 移动开发 ⁄ 共 10555字 ⁄ 字号 评论关闭

异步的轻量级实现; 
AsynceTask简述: 

1.功能类似于Handler,都是为了防止UI线程操作阻塞而衍生而来。 

2.AsyncTask是Handler的一个轻量级实现,模型类似于IntentService于Service。都是为了更加方便操作。(因为一般的异步,我们都是开启一个子线程或是匿名线程,缺点就是样的实现对于线程的操作,控制是十分困难) 

3.阐述下Handler,一般我们就认为Handler既一个Android消息处理器。默认情况下,他只接受当前线程的消息实例。 
但是,当在一个多线程,比如子线程数据处理后更新Ui线程,此时只要存在Handler的指针,简单的说就是实例对象时, 
消息的收发处理就能执行在不同的进程中了,这个也是我们常用到的异步处理手法。 

4.从源代码中看AsyncTask类中有 线程池,同样也实例化了一个Handler对象。 
说白了,AsyncTask只是对以上我们自己用handler,thread实现的异步做了一个很好的封装,使用到线程池对于线程的销毁和创建开销大大减小 

综合了下:AsyncTask的异步处理相对于传统的handler+Thread组合,减少程序中线程过多开销过大。操作和管理更加方便。 

AsyncTask的是实现: 
和所有网上说的一样,该对象必须在UiThread中实例化,然后执行execute方法。 
copy下:AsyncTask定义了三种泛型类型 Params,Progress和Result。 
    •Params 启动任务执行的输入参数,比如HTTP请求的URL。 
    •Progress 后台任务执行的百分比。 
    •Result 后台执行任务最终返回的结果,比如String。 
  
AsyncTask的执行分为四个步骤,每一步都对应一个回调方法,开发者需要实现一个或几个方法。在任务的执行过程中,这些方法被自动调用。 
onPreExecute(), 该方法将在执行实际的后台操作前被UI thread调用。可以在该方法中做一些准备工作,如在界面上显示一个进度条。 
doInBackground(Params...), 将在onPreExecute 方法执行后马上执行,该方法运行在后台线程中。这里将主要负责执行那些很耗时的后台计算工作。可以调用 publishProgress方法来更新实时的任务进度。该方法是抽象方法,子类必须实现。 
onProgressUpdate(Progress...),在publishProgress方法被调用后,UI thread将调用这个方法从而在界面上展示任务的进展情况,例如通过一个进度条进行展示。 
onPostExecute(Result), 在doInBackground 执行完成后,onPostExecute 方法将被UI thread调用,后台的计算结果将通过该方法传递到UI thread. 
  
使用AsyncTask类,以下是几条必须遵守的准则: 
    1) Task的实例必须在UI thread中创建 
    2) execute方法必须在UI thread中调用 
    3) 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)这几个方法 

    4) 该task只能被执行一次,否则多次调用时将会出现异常


下面是我看了源代码书写下整个流程:

step 1:在UiThread中实例化类  执行了父类的构造    

  1. public AsyncTask() {    
  2.     
  3.      mWorker = new WorkerRunnable{//实现线程接扣的线程类。    
  4.          public Result call() {    
  5.              return doInBackground(mParams);//在类似于线程的run方法中调用doInBackground(mParams);    
  6.          }    
  7.      };    
  8.     
  9.      mFuture = new FutureTask(mWorker) {//接受mWorker线程对象,实例化FutureTask类来操作线程。           
  10.  }    

step 2 :在UiThread中调用AsyncTask对象的execute方法。

  1. public execute(){    
  2.            
  3.         onPreExecute();//此方法中先调用了onPreExecute,也就是我们上文提到的准备工作。    
  4.     
  5.         sExecutor.execute(mFuture);// 然后通过线程池操作构造中我们实例化的runnable对象。    
  6.     
  7.         return this;    


step 3 :根据step 2 可以知道 程序这步执行的应该是mFuture 代码中的done()

  1. protected void done() {    
  2.                 Message message;    
  3.                 Result result = null;    
  4.         // 发送消息Tag和 消息处理结果。消息处理结果又用AsyncTaskResult类封装起来,实例化对象传递进去的result就是上面doInBackground方法执行后的结果。    
  5.                 message = sHandler.obtainMessage(MESSAGE_POST_RESULT,    
  6.                         new AsyncTaskResult<Result>(AsyncTask.this, result));    
  7.                 message.sendToTarget();    
  8.             }    

step 4: 根据step 3,很明显 handler发送,后紧接着就是消息的处理

  1. private static class InternalHandler extends Handler {    
  2.          
  3.        @Override    
  4.        public void handleMessage(Message msg) {    
  5.            AsyncTaskResult result = (AsyncTaskResult) msg.obj;    
  6.            switch (msg.what) {    
  7.                case MESSAGE_POST_RESULT://通过step 3的Tag,执行改方法;    
  8.     
  9.                    // There is only one result    
  10.                    result.mTask.finish(result.mData[0]);    
  11.                    break;    
  12.                case MESSAGE_POST_PROGRESS:    
  13.                    result.mTask.onProgressUpdate(result.mData);    
  14.                    break;    
  15.                case MESSAGE_POST_CANCEL:    
  16.                    result.mTask.onCancelled();    
  17.                    break;    
  18.            }    
  19.        }    
  20.    }    

step 5: 根据step 4执行  result.mTask.finish

  1. private void finish(Result result) {    
  2.       if (isCancelled()) result = null;    
  3.       onPostExecute(result);// 注意。。执行了onPostExecute    
  4.       mStatus = Status.FINISHED;    
  5.   }    


至此我们发现上文提到 
onPreExecute()  ----》doInBackground()------》onPostExecute()基本流程方法都游走一边

还剩下两个刷新进度的 :   onProgressUpdate()和publishProgress();

  1. protected final void publishProgress(Progress... values) {    
  2.        sHandler.obtainMessage(MESSAGE_POST_PROGRESS,    
  3.                new AsyncTaskResult<Progress>(this, values)).sendToTarget();    
  4.    }    

从这个方法很容易看出,其实他就是一个Handler消息发送的实现。 
网上一般说可以在doInBackground()调用该方法,然后再回过去看handler实例化那段代码 
 case MESSAGE_POST_PROGRESS: 
                    result.mTask.onProgressUpdate(result.mData); 
                    break; 


执行了onProgressUpdate() 

    该类出现了FutureTask以及callable等java中的类。因为手边没有java  源码。没有细细读下 

    整个流程大致如上。

 

下面附上一个用过的,也算比较常用的首页图片异步获取例子代码:


  1. public class MainActivity extends BaseActivity implements OnClickListener {    
  2.     
  3.     private static final HashMap<String, Drawable> HOMEIMAGECACHE = new HashMap<String, Drawable>();    
  4.     
  5.     @Override    
  6.     public void onCreate(Bundle savedInstanceState) {    
  7.         super.onCreate(savedInstanceState);    
  8.         mContext = MainActivity.this;    
  9.         FrameLayout ll = (FrameLayout) this.getLayoutInflater().inflate(    
  10.                 R.layout.main, null);    
  11.         setContentView(ll);    
  12.         init();    
  13.         fatchData();    
  14.             
  15.     }    
  16.     
  17.     private void fatchData() {    
  18.         GetHomePageImageTask imageTask = new GetHomePageImageTask();    
  19.         //首页图片获取测试    
  20.         String imgUrl = "http://192.168.1.1/img/xxxxx.png,http://192.168.1.1/img/zuqiu2.png,http://192.168.1.1/img/zuqiu3.png";    
  21.         imageTask.execute(imgUrl);      
  22.     }    
  23.     private void init() {    
  24.         loadView();    
  25.             
  26.     }    
  27.     
  28.     private Gallery mGallery;    
  29.         
  30.     /**   
  31.      *    
  32.      */    
  33.     private void loadView() {    
  34.             
  35.         mGallery = (Gallery) findViewById(R.id.homepage_gallery);    
  36.     }    
  37.     
  38.     class GetHomePageImageTask extends AsyncTask<String, Integer, Drawable []>{    
  39.     
  40.         /**   
  41.          * step 1: 在ui Thread中调用execute()后执行该方法;   
  42.          * 一般可以做些准备工作。   
  43.          */    
  44.         @Override     
  45.         protected void onPreExecute() {    
  46.             // TODO Auto-generated method stub    
  47.             super.onPreExecute();    
  48.         }    
  49.             
  50.          /**   
  51.           * step 2: 执行后台操作;   
  52.           */    
  53.             
  54.         @Override    
  55.         protected Drawable[] doInBackground(String... params) {    
  56.                 
  57.         String [] imageUrlsparams[0].split(",");    
  58.         Drawable [] drawables = new Drawable[imageUrls.length];    
  59.         for(int i = 0; i < imageUrls.length;i++ ){    
  60.             drawables[i] = lookupFile(imageUrls[i]);    
  61.         }    
  62.             return drawables;    
  63.         }    
  64.             
  65.         /**   
  66.          *在publishProgress方法被调用后,ui线程将调用该方法更新进度;   
  67.          */    
  68.         @Override    
  69.         protected void onProgressUpdate(Integer... values) {    
  70.             // TODO Auto-generated method stub    
  71.             super.onProgressUpdate(values);    
  72.                 
  73.         }    
  74.             
  75.         /**   
  76.          * step 3:在doInBackground执行完成后i,该方法被Ui线程执行那个调用;   
  77.          */    
  78.         @Override    
  79.         protected void onPostExecute(Drawable[] result) {    
  80.             // TODO Auto-generated method stub    
  81.             super.onPostExecute(result);    
  82.             ImageAdapter adapter = new ImageAdapter(mContext, result);    
  83.             mGallery.setAdapter(adapter);    
  84.         }    
  85.     }    
  86.     class ImageAdapter extends BaseAdapter {    
  87.     
  88.         Holder holder;    
  89.         /* 声明变量 */    
  90.         int mGalleryItemBackground;    
  91.         private Context mContext;    
  92.     
  93.         private Drawable[] _bitmap;    
  94.     
  95.         LayoutInflater inflater;    
  96.     
  97.         /* ImageAdapter的构造器 */    
  98.         public ImageAdapter(Context c, Drawable[] bitmap) {    
  99.             mContext = c;    
  100.             _bitmap = bitmap;    
  101.                 
  102.         }    
  103.     
  104.         /* 覆盖的方法getCount,返回图片数目 */    
  105.         public int getCount() {    
  106.             return _bitmap.length;    
  107.         }    
  108.     
  109.         /* 覆盖的方法getItemId,返回图像的数组id */    
  110.     
  111.         public Object getItem(int position) {    
  112.             return position;    
  113.         }    
  114.     
  115.         public long getItemId(int position) {    
  116.             return position;    
  117.         }    
  118.     
  119.         /* 覆盖的方法getView,返回一View对象 */    
  120.         public View getView(int position, View convertView, ViewGroup parent) {    
  121.             inflater = LayoutInflater.from(mContext);    
  122.             if (convertView == null) {    
  123.                 convertView = inflater.inflate(R.layout.game_detail_item, null);    
  124.                 holder = new Holder();    
  125.                 holder.img = (ImageView) convertView    
  126.                         .findViewById(R.id.game_gallery_img);    
  127.                 convertView.setTag(holder);    
  128.             } else {    
  129.                 holder = (Holder) convertView.getTag();    
  130.             }    
  131.             holder.img.setBackgroundDrawable(_bitmap[position]);    
  132.             return convertView;    
  133.         }    
  134.     
  135.         final class Holder {    
  136.             ImageView img;    
  137.         }    
  138.     }    
  139.     /*   
  140.      * 图片数据读取   
  141.      */    
  142.     public Drawable lookupFile(String url) {    
  143.         // TODO Auto-generated method stub    
  144.         FileInputStream fis = null;    
  145.         Drawable drawable = null;    
  146.         try {    
  147.             if (HOMEIMAGECACHE.get(url) == null) {    
  148.                 URL u = new URL(url);    
  149.                 URLConnection openConnection = u.openConnection();    
  150.                 drawable = Drawable.createFromStream(    
  151.                         openConnection.getInputStream(), "");    
  152.                 drawable.setBounds(0, 0, drawable.getIntrinsicWidth(),    
  153.                         drawable.getIntrinsicHeight());    
  154.                 HOMEIMAGECACHE.put(url, drawable);    
  155.             }    
  156.             return HOMEIMAGECACHE.get(url);    
  157.         } catch (Exception e) {    
  158.             // Not there.    
  159.             return null;    
  160.         } finally {    
  161.             if (fis != null) {    
  162.                 try {    
  163.                     fis.close();    
  164.                 } catch (IOException e) {    
  165.                         
  166.                 }    
  167.             }    
  168.         }    
  169.     }    
  170. }    



代码分析:

1.url是里包含多个图片的网路地址

2.用了一个gallery来显示图片

3.实现自定义的GetHomePageImageTask

4.lookupFile()中用了下缓存,可以忽略。

主要代码还是GetHomePageImageTask中的

  1. /**   
  2.       * step 2: 执行后台操作;   
  3.       */    
  4.         
  5.     @Override    
  6.     protected Drawable[] doInBackground(String... params) {    
  7.             
  8.     String [] imageUrlsparams[0].split(",");    
  9.     Drawable [] drawables = new Drawable[imageUrls.length];    
  10.     for(int i = 0; i < imageUrls.length;i++ ){    
  11.         drawables[i] = lookupFile(imageUrls[i]);    
  12.     }    
  13.         return drawables;    
  14.     }    




  1. /**   
  2.          * step 3:在doInBackground执行完成后i,该方法被Ui线程执行那个调用;   
  3.          */    
  4.         @Override    
  5.         protected void onPostExecute(Drawable[] result) {    
  6.             // TODO Auto-generated method stub    
  7.             super.onPostExecute(result);    
  8.             ImageAdapter adapter = new ImageAdapter(mContext, result);    
  9.             mGallery.setAdapter(adapter);    
  10.         }    


抱歉!评论已关闭.