当前位置:  编程技术>Android网络共享软件 Android Wifi Tether iis7站长之家

Android开发笔记之:深入理解多线程AsyncTask

    来源: 互联网  发布时间:2014-10-16

    本文导语:  Understanding AsyncTaskAsyncTask是Android 1.5 Cubake加入的用于实现异步操作的一个类,在此之前只能用Java SE库中的Thread来实现多线程异步,AsyncTask是Android平台自己的异步工具,融入了Android平台的特性,让异步操作更加的安全,方便和...

Understanding AsyncTask
AsyncTask是Android 1.5 Cubake加入的用于实现异步操作的一个类,在此之前只能用Java SE库中的Thread来实现多线程异步,AsyncTask是Android平台自己的异步工具,融入了Android平台的特性,让异步操作更加的安全,方便和实用。实质上它也是对Java SE库中Thread的一个封装,加上了平台相关的特性,所以对于所有的多线程异步都强烈推荐使用AsyncTask,因为它考虑,也融入了Android平台的特性,更加的安全和高效。
AsyncTask可以方便的执行异步操作(doInBackground),又能方便的与主线程进行通信,它本身又有良好的封装性,可以进行取消操作(cancel())。关于AsyncTask的使用,文档说的很明白,下面直接上实例。
实例
这个实例用AsyncTask到网络上下载图片,同时显示进度,下载完图片更新UI。
代码如下:

package com.hilton.effectiveandroid.concurrent;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import android.app.Activity;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.SystemClock;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.ProgressBar;
import com.hilton.effectiveandroid.R;
/*
 * AsyncTask cannot be reused, i.e. if you have executed one AsyncTask, you must discard it, you cannot execute it again.
 * If you try to execute an executed AsyncTask, you will get "java.lang.IllegalStateException: Cannot execute task: the task is already running"
 * In this demo, if you click "get the image" button twice at any time, you will receive "IllegalStateException".
 * About cancellation:
 * You can call AsyncTask#cancel() at any time during AsyncTask executing, but the result is onPostExecute() is not called after
 * doInBackground() finishes, which means doInBackground() is not stopped. AsyncTask#isCancelled() returns true after cancel() getting
 * called, so if you want to really cancel the task, i.e. stop doInBackground(), you must check the return value of isCancelled() in
 * doInBackground, when there are loops in doInBackground in particular.
 * This is the same to Java threading, in which is no effective way to stop a running thread, only way to do is set a flag to thread, and check
 * the flag every time in Thread#run(), if flag is set, run() aborts.
 */
public class AsyncTaskDemoActivity extends Activity {
    private static final String ImageUrl = "http://i1.cqnews.net/sports/attachement/jpg/site82/2011-10-01/2960950278670008721.jpg";
    private ProgressBar mProgressBar;
    private ImageView mImageView;
    private Button mGetImage;
    private Button mAbort;

    @Override
    public void onCreate(Bundle icicle) {
 super.onCreate(icicle);
 setContentView(R.layout.async_task_demo_activity);
 mProgressBar = (ProgressBar) findViewById(R.id.async_task_progress);
 mImageView = (ImageView) findViewById(R.id.async_task_displayer);
 final ImageLoader loader = new ImageLoader();
 mGetImage = (Button) findViewById(R.id.async_task_get_image);
 mGetImage.setOnClickListener(new View.OnClickListener() {
     public void onClick(View v) {
  loader.execute(ImageUrl);
     }
 });
 mAbort = (Button) findViewById(R.id.asyc_task_abort);
 mAbort.setOnClickListener(new View.OnClickListener() {
     public void onClick(View v) {
  loader.cancel(true);
     }
 });
 mAbort.setEnabled(false);
    }

    private class ImageLoader extends AsyncTask {
 private static final String TAG = "ImageLoader";
 @Override
 protected void onPreExecute() {
     // Initialize progress and image
     mGetImage.setEnabled(false);
     mAbort.setEnabled(true);
     mProgressBar.setVisibility(View.VISIBLE);
     mProgressBar.setProgress(0);
     mImageView.setImageResource(R.drawable.icon);
 }

 @Override
 protected Bitmap doInBackground(String... url) {
     /*
      * Fucking ridiculous thing happened here, to use any Internet connections, either via HttpURLConnection
      * or HttpClient, you must declare INTERNET permission in AndroidManifest.xml. Otherwise you will get
      * "UnknownHostException" when connecting or other tcp/ip/http exceptions rather than "SecurityException"
      * which tells you need to declare INTERNET permission.
      */
     try {
  URL u;
  HttpURLConnection conn = null;
  InputStream in = null;
  OutputStream out = null;
  final String filename = "local_temp_image";
  try {
      u = new URL(/tech-mobile-dev/url[0]/index.html);
      conn = (HttpURLConnection) u.openConnection();
      conn.setDoInput(true);
      conn.setDoOutput(false);
      conn.setConnectTimeout(20 * 1000);
      in = conn.getInputStream();
      out = openFileOutput(filename, Context.MODE_PRIVATE);
      byte[] buf = new byte[8196];
      int seg = 0;
      final long total = conn.getContentLength();
      long current = 0;
      /*
       * Without checking isCancelled(), the loop continues until reading whole image done, i.e. the progress
       * continues go up to 100. But onPostExecute() will not be called.
       * By checking isCancelled(), we can stop immediately, i.e. progress stops immediately when cancel() is called.
       */
      while (!isCancelled() && (seg = in.read(buf)) != -1) {
   out.write(buf, 0, seg);
   current += seg;
   int progress = (int) ((float) current / (float) total * 100f);
   publishProgress(progress);
   SystemClock.sleep(1000);
      }
  } finally {
      if (conn != null) {
   conn.disconnect();
      }
      if (in != null) {
   in.close();
      }
      if (out != null) {
   out.close();
      }
  }
  return BitmapFactory.decodeFile(getFileStreamPath(filename).getAbsolutePath());
     } catch (MalformedURLException e) {
  e.printStackTrace();
     } catch (IOException e) {
  e.printStackTrace();
     }
     return null;
 }

 @Override
 protected void onProgressUpdate(Integer... progress) {
     mProgressBar.setProgress(progress[0]);
 }

 @Override
 protected void onPostExecute(Bitmap image) {
     if (image != null) {
  mImageView.setImageBitmap(image);
     }
     mProgressBar.setProgress(100);
     mProgressBar.setVisibility(View.GONE);
     mAbort.setEnabled(false);
 }
    }
}

运行结果

先后顺序分别是下载前,下载中和下载后
总结
关于怎么使用看文档和这个例子就够了,下面说下,使用时的注意事项:
1. AsyncTask对象不可重复使用,也就是说一个AsyncTask对象只能execute()一次,否则会有异常抛出"java.lang.IllegalStateException: Cannot execute task: the task is already running"

2. 在doInBackground()中要检查isCancelled()的返回值,如果你的异步任务是可以取消的话。
cancel()仅仅是给AsyncTask对象设置了一个标识位,当调用了cancel()后,发生的事情只有:AsyncTask对象的标识位变了,和doInBackground()执行完成后,onPostExecute()不会被回调了,而doInBackground()和onProgressUpdate()还是会继续执行直到doInBackground()结束。所以要在doInBackground()中不断的检查isCancellled()的返回值,当其返回true时就停止执行,特别是有循环的时候。如上面的例子,如果把读取数据的isCancelled()检查去掉,图片还是会下载,进度也一直会走,只是最后图片不会放到UI上(因为onPostExecute()没被回调)!

这里的原因其实很好理解,想想Java SE的Thread吧,是没有方法将其直接Cacncel掉的,那些线程取消也无非就是给线程设置标识位,然后在run()方法中不断的检查标识而已。

3. 如果要在应用程序中使用网络,一定不要忘记在AndroidManifest中声明INTERNET权限,否则会报出很诡异的异常信息,比如上面的例子,如果把INTERNET权限拿掉会抛出"UnknownHostException"。刚开始很疑惑,因为模拟器是可以正常上网的,后来Google了下才发现原来是没权限,但是疑问还是没有消除,既然没有声明网络权限,为什么不直接提示无网络权限呢?

对比Java SE的Thread
Thread是非常原始的类,它只有一个run()方法,一旦开始,无法停止,它仅适合于一个非常独立的异步任务,也即不需要与主线程交互,对于其他情况,比如需要取消或与主线程交互,都需添加额外的代码来实现,并且还要注意同步的问题。
而AsyncTask是封装好了的,可以直接拿来用,如果你仅执行独立的异步任务,可以仅实现doInBackground()。
所以,当有一个非常独立的任务时,可以考虑使用Thread,其他时候,尽可能的用AsyncTask。

    
 
 

您可能感兴趣的文章:

  • android使用handler ui线程和子线程通讯更新ui示例
  • Android 在其他线程中更新UI线程的解决方法
  • Android开发笔记之:如何安全中止一个自定义线程Thread的方法
  • Handler与Android多线程详解
  • 浅析android中的线程封装
  • Android Handler主线程和一般线程通信的应用分析
  • android开发教程之子线程中更新界面
  • android使用多线程更新ui示例分享
  • android使用handlerthread创建线程示例
  • Android多线程处理机制中的Handler使用介绍
  • Android后台线程和UI线程通讯实例
  • Android提高之SurfaceView与多线程的混搭实例
  • 在Android线程池里运行代码任务实例
  • android开发教程之handle实现多线程和异步处理
  • Android中创建多线程管理器实例
  • android开发教程之使用线程实现视图平滑滚动示例
  • Android多线程及异步处理问题详细探讨
  • 深入Android线程的相关问题解惑
  • 深入Android Handler与线程间通信ITC的详解
  • android 多线程技术应用
  • android教程之使用asynctask在后台运行耗时任务
  • Android中AsyncTask的用法实例分享
  • Android开发笔记之:AsyncTask的应用详解
  • android异步任务设计思详解(AsyncTask)
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • 深入android Unable to resolve target 'android-XX'详解
  • 深入Android开发FAQ的详解
  • Android 自动化测试经验分享 深入UiScrollable
  • 基于android中读取assets目录下a.txt文件并进行解析的深入分析
  • Android中asset文件夹与raw文件夹的区别深入解析
  • Android开发之文件操作模式深入理解
  • Android Mms之:深入理解对话列表管理
  • android生命周期深入分析(一)
  • Android 关机弹出选择菜单的深入解析
  • 深入Android SQLite 事务处理详解
  • ubuntu 12.10 上 android 编译环境搭建的深入解析
  • 深入Android Browser配置管理的详解
  • 深入Android MediaPlayer的使用方法详解
  • Android中使用PULL方式解析XML文件深入介绍
  • 深入android中The connection to adb is down的问题以及解决方法
  • Android笔记之:深入为从右向左语言定义复杂字串的详解
  • Android开发笔记之:深入理解Cursor相关的性能问题
  • 深入Understanding Android ContentProvider详解
  • Android笔记之:深入ViewStub的应用
  • 深入分析Android ViewStub的应用详解
  • 申请Android Map 的API Key(v2)的最新申请方式(SHA1密钥)
  • Android瀑布流实例 android_waterfall
  • Android开发需要的几点注意事项总结
  • Android系统自带样式 (android:theme)
  • android 4.0 托管进程介绍及优先级和回收机制
  • Android网络共享软件 Android Wifi Tether
  • Android访问与手机通讯相关类的介绍
  • Android 图标库 Android GraphView
  • Android及andriod无线网络Wifi开发的几点注意事项
  • 轻量级Android开发工具 Android Tools
  • Android 2.3 下StrictMode介绍


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3