AsyncTask的线程池使用

这几天写了一个demo,利用asynctask来异步加载手机相册的照片,小米2S,4.1.1,但是发现相片是一张接一张加载出来的,而不是好几张同时加载的,说明我起的那些异步的线程在线程池里面排起队了,一个队,一个一个服务。

于是在网上搜了一下,才发现asynctask的线程池机制是跟系统版本有关系的,参考http://blog.csdn.net/hitlion2008/article/details/7983449

大致就是2.3以前的版本asynctask的线程池默认是Executors.newFixedThreadPool(5),即是5个服务的线程池,而3.0之后是由一个队列来维护的单队伍的线程池。

/**
     * An {@link Executor} that can be used to execute tasks in parallel.
     */
    public static final Executor THREAD_POOL_EXECUTOR
            = new ThreadPoolExecutor(CORE_POOL_SIZE, MAXIMUM_POOL_SIZE, KEEP_ALIVE,
                    TimeUnit.SECONDS, sPoolWorkQueue, sThreadFactory);

    /**
     * An {@link Executor} that executes tasks one at a time in serial
     * order.  This serialization is global to a particular process.
     */
    public static final Executor SERIAL_EXECUTOR = new SerialExecutor();
一个并行,一个串行。

private static volatile Executor sDefaultExecutor = SERIAL_EXECUTOR;
3.0之后默认的是串行的。

    private static class SerialExecutor implements Executor {
        final ArrayDeque<Runnable> mTasks = new ArrayDeque<Runnable>();
        Runnable mActive;

        public synchronized void execute(final Runnable r) {
            mTasks.offer(new Runnable() {
                public void run() {
                    try {
                        r.run();
                    } finally {
                        scheduleNext();
                    }
                }
            });
            if (mActive == null) {
                scheduleNext();
            }
        }

        protected synchronized void scheduleNext() {
            if ((mActive = mTasks.poll()) != null) {
                THREAD_POOL_EXECUTOR.execute(mActive);
            }
        }
    }
串行线程池实现,就是维护了一个队列,等队头执行完了,再执行下一个,而干苦力的依然是并行的线程池
THREAD_POOL_EXECUTOR



所幸,asynctask提供了一个用户自己来指定线程池的方法

public final AsyncTask<Params, Progress, Result> executeOnExecutor(Executor exec,
            Params... params) {
        if (mStatus != Status.PENDING) {
            switch (mStatus) {
                case RUNNING:
                    throw new IllegalStateException("Cannot execute task:"
                            + " the task is already running.");
                case FINISHED:
                    throw new IllegalStateException("Cannot execute task:"
                            + " the task has already been executed "
                            + "(a task can be executed only once)");
            }
        }

        mStatus = Status.RUNNING;

        onPreExecute();

        mWorker.mParams = params;
        exec.execute(mFuture);

        return this;
    }
所以使用的时候可以使用THREAD_POOL_EXECUTOR,也可以使用自己定义的线程池,如

private ExecutorService mExecutor = Executors.newFixedThreadPool(5);
    ……
private void loaddata(){
    ……
    task.executeOnExecutor(mExecutor, imageUrl);
}



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
异步任务(AsyncTask)和线程池都是将网络操作放到子线程中执行的常用方式。 1. AsyncTask方式 使用AsyncTask时需要重写其doInBackground()方法,在其中执行耗时操作,例如网络请求。然后可以在onPostExecute()方法中更新UI界面。 下面是一个使用AsyncTask进行网络请求的例子: ``` public class MyAsyncTask extends AsyncTask<String, Void, String> { @Override protected String doInBackground(String... params) { String url = params[0]; String result = ""; try { URL urlObj = new URL(url); HttpURLConnection conn = (HttpURLConnection) urlObj.openConnection(); conn.setRequestMethod("GET"); conn.setConnectTimeout(5000); conn.setReadTimeout(5000); if (conn.getResponseCode() == 200) { InputStream is = conn.getInputStream(); ByteArrayOutputStream bos = new ByteArrayOutputStream(); byte[] buffer = new byte[1024]; int len; while ((len = is.read(buffer)) != -1) { bos.write(buffer, 0, len); } result = new String(bos.toByteArray()); bos.close(); is.close(); } conn.disconnect(); } catch (Exception e) { e.printStackTrace(); } return result; } @Override protected void onPostExecute(String result) { // 在这里更新UI界面 super.onPostExecute(result); } } ``` 2. 线程池方式 线程池是一个管理线程的容器,可以创建多个线程,将任务分配给这些线程执行。这样可以避免频繁地创建和销毁线程,提高了线程的利用率。 下面是一个使用线程池进行网络请求的例子: ``` public class MyThreadPool { private static ExecutorService executorService = Executors.newFixedThreadPool(5); public static void execute(Runnable runnable) { executorService.execute(runnable); } } public class MyRunnable implements Runnable { private String url; public MyRunnable(String url) { this.url = url; } @Override public void run() { String result = ""; try { URL urlObj = new URL(url); HttpURLConnection conn = (HttpURLConnection) urlObj.openConnection(); conn.setRequestMethod("GET"); conn.setConnectTimeout(5000); conn.setReadTimeout(5000); if (conn.getResponseCode() == 200) { InputStream is = conn.getInputStream(); ByteArrayOutputStream bos = new ByteArrayOutputStream(); byte[] buffer = new byte[1024]; int len; while ((len = is.read(buffer)) != -1) { bos.write(buffer, 0, len); } result = new String(bos.toByteArray()); bos.close(); is.close(); } conn.disconnect(); } catch (Exception e) { e.printStackTrace(); } // 在这里更新UI界面 } } ``` 使用线程池时,需要先创建一个ExecutorService对象,然后通过execute()方法提交任务。在任务的run()方法中执行网络请求,并在其中更新UI界面。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值