android异步类AsyncTask详解

上一篇博客中介绍了Android 的Handler消息机制,现在我们来了解Android中的另一个异步类AsyncTask。

AsyncTask,是android提供的轻量级的异步类,可以直接继承AsyncTask,在类中实现异步操作,并提供接口反馈当前异步执行的程度(可以通过接口实现UI进度更新),最后反馈执行的结果给UI主线程.
AsyncTask底层其实也是通过Handler来完成的。
废话不说,直接上代码:

new AsyncTask<String,String,String>(){
        // 运行在主线程中,做预备工作,当用户开始执行任务时调用次方法
        onPreExecute(){

        }
        // 运行在子线程中,做耗时操作
        String doingBackGround(String s){

        }
        // 运行在主线程中,耗时操作完成,更新UI
        onPostExecute(String s){

        }

    }.execute(String);

上面的方法是很常用的。其实AsyncTask类还有如下的方法。如果需要重写即可:

    //可以使用进度条增加用户体验度。 此方法在主线程执行,用于显示任务执行的进度。
    onProgressUpdate(Progress…)   
    //用户调用取消时,要做的操作
    onCancelled()             

使用AsyncTask类,以下是几条必须遵守的准则:

  • Task的实例必须在UI thread中创建;
  • execute方法必须在UI thread中调用;
  • 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params…),onProgressUpdate(Progress…)这几个方法;
  • 该task只能被执行一次,否则多次调用时将会出现异常;

实例:(下载操作):
AsyncTask的三个参数的意义:
1.第一个参数是方法doingBackGround()的参数,它的返回值对应第三个参数也是方法onPostExecute()的参数
2.第二个参数是onProgressUpdate()方法的参数,它由doingBackGround()方法调用publishProgress()作为实参传入。

这里写图片描述

这里写图片描述

AsyncTask的execute方法:

public final AsyncTask<Params, Progress, Result> execute(Params... params) {
            ...

            mStatus = Status.RUNNING;
            // 在主线程中执行准备操作
            onPreExecute();
            // 把params参数赋值给mWorker
            mWorker.mParams = params;
            // 用线程池执行mFuture
            sExecutor.execute(mFuture);

    // 在AsyncTask构造方法中创建了mWorker
        mWorker = new WorkerRunnable<Params, Result>() {
            public Result call() throws Exception {
               ...
            }
        };

        mFuture = new FutureTask<Result>(mWorker) {
            @Override
            protected void done() {
                ...
            }
        };
    }
        // 把mWorker传递给FutureTask,callable指的就是mWorker
        public FutureTask(Callable<V> callable) {
                if (callable == null)
                    throw new NullPointerException();
                sync = new Sync(callable);
            }
        // 把mWorker传递给Sync,callable指的是mWorker
        Sync(Callable<V> callable) {
                    this.callable = callable;
                }

线程池执行FutureTask,就是执行FutureTask的run方法,代码如下:

    public void run() {
                // 转调
                sync.innerRun();
            }

        void innerRun() {
                    if (!compareAndSetState(READY, RUNNING))
                        return;

                    runner = Thread.currentThread();
                    if (getState() == RUNNING) { // recheck after setting thread
                        V result;
                        try {
                            // 就是调用了mWorker.call方法
                            // 把耗时操作得到的结果赋值给result
                            result = callable.call();
                        } catch (Throwable ex) {
                            setException(ex);
                            return;
                        }
                        // 转调了sync.innerSet(v);
                        set(result);
                    } else {
                        releaseShared(0); // cancel
                    }
                }

        mWorker = new WorkerRunnable<Params, Result>() {
                    public Result call() throws Exception {
                        Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);
                        // 执行耗时操作 在子线程中执行
                        return doInBackground(mParams);
                    }
                };

        protected void set(V v) {
                // 转调
                sync.innerSet(v);
            }

        void innerSet(V v) {
                    for (;;) {
                        int s = getState();
                        if (s == RAN)
                            return;
                        if (s == CANCELLED) {
                            // aggressively release to set runner to null,
                            // in case we are racing with a cancel request
                            // that will try to interrupt runner
                            releaseShared(0);
                            return;
                        }
                        if (compareAndSetState(s, RAN)) {
                            result = v;
                            releaseShared(0);
                            // 调用了FutureTask的抽象方法
                            done();
                            return;
                        }
                    }
                }

        mFuture = new FutureTask<Result>(mWorker) {
            @Override
            protected void done() {
                Message message;
                Result result = null;

                try {
                    // 转调了sync.innerGet()
                    result = get();
                } catch (InterruptedException e) {
                    android.util.Log.w(LOG_TAG, e);
                } catch (ExecutionException e) {
                    throw new RuntimeException("An error occured while executing doInBackground()",
                            e.getCause());
                } catch (CancellationException e) {
                    message = sHandler.obtainMessage(MESSAGE_POST_CANCEL,
                            new AsyncTaskResult<Result>(AsyncTask.this, (Result[]) null));
                    message.sendToTarget();
                    return;
                } catch (Throwable t) {
                    throw new RuntimeException("An error occured while executing "
                            + "doInBackground()", t);
                }
                // 发送了一个Message
                message = sHandler.obtainMessage(MESSAGE_POST_RESULT,
                        new AsyncTaskResult<Result>(AsyncTask.this, result));
                message.sendToTarget();
            }
        };

        public V get() throws InterruptedException, ExecutionException {
                // 转调
                return sync.innerGet();
            }

        V innerGet() throws InterruptedException, ExecutionException {
                    acquireSharedInterruptibly(0);
                    if (getState() == CANCELLED)
                        throw new CancellationException();
                    if (exception != null)
                        throw new ExecutionException(exception);
                    // 把之前doinBackground方法的结果返回
                    return result;
                }

在AsyncTask的成员变量,创建了InternalHandler

 private static class InternalHandler extends Handler {
        @SuppressWarnings({"unchecked", "RawUseOfParameterizedType"})
        @Override
        public void handleMessage(Message msg) {
            AsyncTaskResult result = (AsyncTaskResult) msg.obj;
            switch (msg.what) {
                case MESSAGE_POST_RESULT:
                    // There is only one result
                    // 结束耗时操作完成后的消息
                    // 调用了AsyncTask的finish方法传递的result.mData[0]就是之前
                    // 耗时操作返回来的结果
                    result.mTask.finish(result.mData[0]);
                    break;
                case MESSAGE_POST_PROGRESS:
                    result.mTask.onProgressUpdate(result.mData);
                    break;
                case MESSAGE_POST_CANCEL:
                    result.mTask.onCancelled();
                    break;
            }
        }
    }

    private static class AsyncTaskResult<Data> {
        final AsyncTask mTask;
        final Data[] mData;
        // data 是返回的结果
        AsyncTaskResult(AsyncTask task, Data... data) {
            mTask = task;
            mData = data;
        }
    }

    private void finish(Result result) {
        if (isCancelled()) result = null;
        // 耗时操作完成,更新UI,执行在主线程
        onPostExecute(result);
        mStatus = Status.FINISHED;
    }
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值