Android 异步操作AsyncTask

•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中实例化类  执行了父类的构造    ( 代码都简写了)

[java]  view plain  copy

  1. public AsyncTask() {

  2. mWorker = new WorkerRunnable{//实现线程接扣的线程类。

  3. public Result call() {

  4. return doInBackground(mParams);//在类似于线程的run方法中调用doInBackground(mParams);

  5. }

  6. };

  7. mFuture = new FutureTask(mWorker) {//接受mWorker线程对象,实例化FutureTask类来操作线程。

  8. }

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

[java]  view plain  copy

  1. public execute(){

  2. onPreExecute();//此方法中先调用了onPreExecute,也就是我们上文提到的准备工作。

  3. sExecutor.execute(mFuture);// 然后通过线程池操作构造中我们实例化的runnable对象。

  4. return this;

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

[java]  view plain  copy

  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(AsyncTask.this, result));

  7. message.sendToTarget();

  8. }

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

[java]  view plain  copy

  1. private static class InternalHandler extends Handler {

  2. @Override

  3. public void handleMessage(Message msg) {

  4. AsyncTaskResult result = (AsyncTaskResult) msg.obj;

  5. switch (msg.what) {

  6. case MESSAGE_POST_RESULT://通过step 3的Tag,执行改方法;

  7. // There is only one result

  8. result.mTask.finish(result.mData[0]);

  9. break;

  10. case MESSAGE_POST_PROGRESS:

  11. result.mTask.onProgressUpdate(result.mData);

  12. break;

  13. case MESSAGE_POST_CANCEL:

  14. result.mTask.onCancelled();

  15. break;

  16. }

  17. }

  18. }

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

[java]  view plain  copy

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

[java]  view plain  copy

  1. protected final void publishProgress(Progress… values) {

  2. sHandler.obtainMessage(MESSAGE_POST_PROGRESS,

  3. new AsyncTaskResult(this, values)).sendToTarget();

  4. }

从这个方法很容易看出,其实他就是一个Handler消息发送的实现。

网上一般说可以在doInBackground()调用该方法,然后再回过去看handler实例化那段代码

case MESSAGE_POST_PROGRESS:

result.mTask.onProgressUpdate(result.mData);

break;

执行了onProgressUpdate()

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

整个流程大致如上。

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

[java]  view plain  copy

  1. public class MainActivity extends BaseActivity implements OnClickListener {

  2. private static final HashMap<String, Drawable> HOMEIMAGECACHE = new HashMap<String, Drawable>();

  3. @Override

  4. public void onCreate(Bundle savedInstanceState) {

  5. super.onCreate(savedInstanceState);

  6. mContext = MainActivity.this;

  7. FrameLayout ll = (FrameLayout) this.getLayoutInflater().inflate(

  8. R.layout.main, null);

  9. setContentView(ll);

  10. init();

  11. fatchData();

  12. }

  13. private void fatchData() {

  14. GetHomePageImageTask imageTask = new GetHomePageImageTask();

  15. //首页图片获取测试

  16. 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”;

  17. imageTask.execute(imgUrl);

  18. }

  19. private void init() {

  20. loadView();

  21. }

  22. private Gallery mGallery;

  23. /**

  24. *

  25. */

  26. private void loadView() {

  27. mGallery = (Gallery) findViewById(R.id.homepage_gallery);

  28. }

  29. class GetHomePageImageTask extends AsyncTask<String, Integer, Drawable []>{

  30. /**

  31. * step 1: 在ui Thread中调用execute()后执行该方法;

  32. * 一般可以做些准备工作。

  33. */

  34. @Override

  35. protected void onPreExecute() {

  36. // TODO Auto-generated method stub

  37. super.onPreExecute();

  38. }

  39. /**

  40. * step 2: 执行后台操作;

  41. */

  42. @Override

  43. protected Drawable[] doInBackground(String… params) {

  44. String [] imageUrls= params[0].split(“,”);

  45. Drawable [] drawables = new Drawable[imageUrls.length];

  46. for(int i = 0; i < imageUrls.length;i++ ){

  47. drawables[i] = lookupFile(imageUrls[i]);

  48. }

  49. return drawables;

  50. }

  51. /**

  52. *在publishProgress方法被调用后,ui线程将调用该方法更新进度;

  53. */

  54. @Override

  55. protected void onProgressUpdate(Integer… values) {

  56. // TODO Auto-generated method stub

  57. super.onProgressUpdate(values);

  58. }

  59. /**

  60. * step 3:在doInBackground执行完成后i,该方法被Ui线程执行那个调用;

  61. */

  62. @Override

  63. protected void onPostExecute(Drawable[] result) {

  64. // TODO Auto-generated method stub

  65. super.onPostExecute(result);

  66. ImageAdapter adapter = new ImageAdapter(mContext, result);

  67. mGallery.setAdapter(adapter);

  68. }

  69. }

  70. class ImageAdapter extends BaseAdapter {

  71. Holder holder;

  72. /* 声明变量 */

  73. int mGalleryItemBackground;

  74. private Context mContext;

  75. private Drawable[] _bitmap;

  76. LayoutInflater inflater;

  77. /* ImageAdapter的构造器 */

  78. public ImageAdapter(Context c, Drawable[] bitmap) {

  79. mContext = c;

  80. _bitmap = bitmap;

  81. }

  82. /* 覆盖的方法getCount,返回图片数目 */

  83. public int getCount() {

  84. return _bitmap.length;

  85. }

  86. /* 覆盖的方法getItemId,返回图像的数组id */

  87. public Object getItem(int position) {

  88. return position;

  89. }

  90. public long getItemId(int position) {

  91. return position;

  92. }

  93. /* 覆盖的方法getView,返回一View对象 */

  94. public View getView(int position, View convertView, ViewGroup parent) {

  95. inflater = LayoutInflater.from(mContext);

  96. if (convertView == null) {

  97. convertView = inflater.inflate(R.layout.game_detail_item, null);

  98. holder = new Holder();

  99. holder.img = (ImageView) convertView

  100. .findViewById(R.id.game_gallery_img);

  101. convertView.setTag(holder);

  102. } else {

  103. holder = (Holder) convertView.getTag();

  104. }

  105. holder.img.setBackgroundDrawable(_bitmap[position]);

  106. return convertView;

  107. }

  108. final class Holder {

  109. ImageView img;

  110. }

  111. }

  112. /*

  113. * 图片数据读取

  114. */

  115. public Drawable lookupFile(String url) {

  116. // TODO Auto-generated method stub

  117. FileInputStream fis = null;

  118. Drawable drawable = null;

  119. try {

  120. if (HOMEIMAGECACHE.get(url) == null) {

  121. URL u = new URL(url);

  122. URLConnection openConnection = u.openConnection();

  123. drawable = Drawable.createFromStream(

  124. openConnection.getInputStream(), “”);

  125. drawable.setBounds(0, 0, drawable.getIntrinsicWidth(),

  126. drawable.getIntrinsicHeight());

  127. HOMEIMAGECACHE.put(url, drawable);

  128. }

  129. return HOMEIMAGECACHE.get(url);

  130. } catch (Exception e) {

  131. // Not there.

  132. return null;

  133. } finally {

  134. if (fis != null) {

  135. try {

  136. fis.close();

  137. } catch (IOException e) {

  138. }

  139. }

  140. }

  141. }

  142. }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值