• Android:异步处理之AsyncTask的应用(二)


    前言

      在上一篇文章中《Android:异步处理之Handler+Thread的应用(一)》,我们知道Android的UI主线程主要负责处理用户的按键事件、用户的触屏事件以及屏幕绘图事件等;既然UI老人家都这么忙了,我们这些开发者肯定不能不识趣的去添乱阻塞UI线程什么的,否则UI界面万一停止响应了呢——这不是招骂的节奏么?!所以我们知道用Handler+Thread的方法,在子线程中处理耗时的任务,任务完成后通过Handler通知UI主线程更新UI界面,皆大欢喜有木有。

      可是这样,还是有某些人觉得用Handler+Thread的代码会比较繁琐,当然这个某些人里面包括我们伟大的谷歌。所以AsyncTask(异步任务)在Android 1.5中横空出世;相对于Handler来说,由于比较好的封装,AsyncTask显得更加轻量级一点,适用于简单的异步处理;当然使用起来也比较简洁,果然是谷歌的亲儿子!

    概述

      AsyncTask是一个抽象类,通常是被继承的命。AsyncTask的内部会维持一个静态的线程池,每个后台任务自然也会被提交到线程池中运行,同时也使用Handler+Thread的机制来调用AsyncTask的各个回调方法;回调方法是在主线程运行的,所以该干什么我们都懂(~ o ~)~zZ(赶紧跟UI界面套近乎呀)。

    我们知道AsyncTask<Params, Progress, Result>是抽象类,我们可以在这里面看出它支持三种泛型:

    1、Params:我们的AsyncTask要开始干活时,我们给他的输入的参数的类型,也就是传递给后台的参数

    2、Progress:AsyncTask向我们报告它干活进度的参数类型,举个例子就是下载进度的百分比

    3、Result:后台执行任务完成,返回的结果的参数类型

    如果某个泛型我们不需要指定,我们可以大大方方的指定Void,没事AsyncTask不会伤心滴。

    当然谷歌也帮我们将AsyncTask的后台任务运行的五种状态,分别是:1、准备运行,2、正在后台运行,3、进度更新,4、完成后台任务,5、取消任务。每种状态在AsyncTask中各有相应的回调方法。

    1、准备运行:onPreExecute(),在任务开启时该回调方法立即在UI线程中被调用,同时也是在执行后台耗时操作前被调用;通常该方法用于完成一些初始化工作,比如在界面上显示进度条等。

    2、正在后台运行:doInBackground(Params...),该回调函数由后台线程在onPreExecute()方法执行结束后立即调用,重写该方法就是后台线程将要完成的耗时任务;由于是由后台线程调用,所以我们不能直接在这里更新UI界面,应该使用publishProgress(Progress...)触发回调方法onProgressUpdate(Progress...)进行进度更新;任务计算的结果必须由该函数返回,并被传递到onPostExecute()中。

    3、进度更新:onProgressUpdate(Progress...),在doInBackground()中调用publishProgress()方法更新任务的执行进度,将会在主线程中触发该方法,一般用于动态地显示一个进度条。

    4、完成后台任务:onPostExecute(Result),当doInBackground()完成后,系统会自动调用onPostExecute()方法,并将doInBackground()的返回值传递给该方法。

    5、取消任务:onCancelled (),在调用AsyncTask的cancel()方法时调用。

    案例

    参考代码:

    public class MainActivity extends ActionBarActivity implements OnClickListener{
    
        private Button startdownload;
        private ProgressBar probar;
        private TextView tv;
        
        private DownTask task;
        
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            
            startdownload = (Button) findViewById(R.id.startdownload);
            probar = (ProgressBar) findViewById(R.id.probar);
            tv = (TextView) findViewById(R.id.tv);
            
            startdownload.setOnClickListener(this);
            
        }
    
        @Override
        public void onClick(View v) {
            task = new DownTask();//同一个AsyncTask的execute只能调用一次
            task.execute("输入参数,可为空");//调用execute后将会回调onPreExecute方法
        }
        
        class DownTask extends AsyncTask<String, Integer, String>{
    
            @Override//该方法非在主线程运行,可进行耗时操作,不可更新UI界面,其他方法为主线程运行
            protected String doInBackground(String... params) {//params为execute输入的参数
                for(int i = 1; i <= 100; i++){
                    try {//模拟下载操作
                        Thread.sleep(333);
                        publishProgress(i);//传递参数i并触发onProgressUpdate回调方法
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                String result = "任务已完成";
                return result;//将调用onPostExecute,并将result传给该回调方法
            }
    
            @Override
            protected void onPreExecute() {//该回调方法执行完毕后,将会调用doInBackground
                probar.setMax(100);
                probar.setProgress(0);
                tv.setText("开始下载");
            }
    
            @Override
            protected void onPostExecute(String result) {//doInBackground结束后回调该方法,结束。
                Toast.makeText(MainActivity.this, result, Toast.LENGTH_SHORT).show();
                tv.setText("下载完成");
            }
    
            @Override
            protected void onProgressUpdate(Integer... values) {//通知UI界面更新
                probar.setProgress(values[0]);
            }
            
        }
    
    }

    布局文件:

    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical" >
        
        <Button
            android:id="@+id/startdownload"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="开始下载"/>
        <ProgressBar 
            android:id="@+id/probar"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            style="@android:style/Widget.ProgressBar.Horizontal"
            />
        <TextView 
            android:id="@+id/tv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textColor="#000000"
            android:textSize="20sp"
            />
    
    </LinearLayout>

    代码讲解:

    1、点击Button后先实例化一个AsyncTask的继承子类,此时将会创建一个task。接下来变执行execute(params)方法启动异步任务。(同一个AsyncTask的实例只能执行execute一次,多次执行会抛出错误)。

    2、在execute()被执行后,将会触发onPreExecute()回调方法,设置进度条的初始属性。在onPreExecute()执行完毕后,将会在后台线程开始执行doInBackground(params),该方法接收execute传入的参数,进行耗时操作,这里是模拟网络文件下载任务。

    3、doInBackground()在后台线程运行中,如果需要与UI主线程交互更新进度,可以调用publishProgress(values)方法,将会触发位于UI主线程运行的onProgressUpdate(values)的回调方法,代码中在这里更新进度条的进度。

    4、 当后台任务执行完成后,调用onPostExecute(Result),传入的参数是doInBackground()中返回的对象。

    注意事项

    1、不要在同一个AsyncTask实例中多次执行execute(),正确的方法是new一个AsyncTask执行一次execute()。

    2、耗时任务一定要在doInBackground()中处理,不要在其他回调方法中处理耗时任务以免引起UI主线程的阻塞。

    3、不要再doInBackground()中更新UI界面,应该通过publishProgress()调用回调方法更新UI。

    4、onCancelled()只能触发AsyncTask的cancel()方法,并无法取消正在线程池运行的线程任务,但可以通过标志位来停止线程任务。

    5、在不同的android版本中,AsyncTask多任务运行,有些是可以并行有些则是顺序执行,不过在高版本Android中,可以通过指定参数设置线程池执行规则。

    6、AsyncTask适合处理短时间的操作,长时间的操作,比如下载一个很大的视频,这就需要你使用自己的线程来下载,不管是断点下载还是其它的。

    作者:enjoy风铃
    出处:http://www.cnblogs.com/net168/
    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则下次不给你转载了。

  • 相关阅读:
    (原创)如何利用UDP协议封装一个数据包
    <acarousel> 轮播图片无法触屏滑动
    Groovy开发语言
    关于ViewStub标签
    关键字transient和Volatile
    Android中Activity启动模式
    Android 开源项目分类汇总
    AtomicInteger的并发处理
    Cygwin: died waiting for dll loading (转载)
    Android知识点(C2DM)
  • 原文地址:https://www.cnblogs.com/net168/p/4080171.html
Copyright © 2020-2023  润新知