Android_AsyncTask详解及其应用(二 )_RejectedExecutionException问题

在这篇文章里面说了AsyncTask的基本用法,想要进一步了解它是如何运行的朋友可以看一下这篇文章:点击,里面详细介绍了AsyncTask的原理以及运行流程。之前只是使用AsyncTask下载一张图片,显然在一个应用中不太现实,如果是很多图片,那使用AsyncTask会不会出现问题呢?下面来试一下,让ListView显示多张图片,使用AsyncTask下载图片。

    代码如下:

[java]  view plain copy
  1. package com.example.asynctaskdemo;  
  2.   
  3. import java.io.ByteArrayOutputStream;  
  4. import java.io.IOException;  
  5. import java.io.InputStream;  
  6. import java.net.HttpURLConnection;  
  7. import java.net.MalformedURLException;  
  8. import java.net.URL;  
  9. import android.app.Activity;  
  10. import android.graphics.Bitmap;  
  11. import android.graphics.BitmapFactory;  
  12. import android.os.AsyncTask;  
  13. import android.os.Bundle;  
  14. import android.view.LayoutInflater;  
  15. import android.view.View;  
  16. import android.view.ViewGroup;  
  17. import android.widget.BaseAdapter;  
  18. import android.widget.ImageView;  
  19. import android.widget.ListView;  
  20.   
  21. public class Asy4Activity extends Activity {  
  22.   
  23.     private ListView listview;  
  24.   
  25.     @Override  
  26.     protected void onCreate(Bundle savedInstanceState) {  
  27.         super.onCreate(savedInstanceState);  
  28.         setContentView(R.layout.activity_handlerimageloader);  
  29.   
  30.         listview = (ListView) findViewById(R.id.listview);  
  31.         listview.setAdapter(new MyAdapter());  
  32.     }  
  33.   
  34.     private class MyAdapter extends BaseAdapter {  
  35.   
  36.         public MyAdapter() {  
  37.   
  38.         }  
  39.   
  40.         @Override  
  41.         public int getCount() {  
  42.             // TODO Auto-generated method stub  
  43.             return 100;  
  44.         }  
  45.   
  46.         @Override  
  47.         public Object getItem(int position) {  
  48.             // TODO Auto-generated method stub  
  49.             return null;  
  50.         }  
  51.   
  52.         @Override  
  53.         public long getItemId(int position) {  
  54.             // TODO Auto-generated method stub  
  55.             return 0;  
  56.         }  
  57.   
  58.         @Override  
  59.         public View getView(int position, View convertView, ViewGroup parent) {  
  60.             if (convertView == null) {  
  61.                 convertView = LayoutInflater.from(getApplicationContext())  
  62.                         .inflate(R.layout.list_item, null);  
  63.             }  
  64.             final ImageView image = (ImageView) convertView  
  65.                     .findViewById(R.id.imageview);  
  66.             String imageURL = "http://images.ladymax.cn/userup/1305/29111RO915.jpg";  
  67.   
  68.             class AsyncTaskLoadImage extends AsyncTask<String, Integer, Bitmap> {  
  69.   
  70.                 @Override  
  71.                 protected void onPreExecute() {  
  72.                     super.onPreExecute();  
  73.                 }  
  74.   
  75.                 @Override  
  76.                 protected Bitmap doInBackground(String... params) {  
  77.                     Bitmap bitmap = null;  
  78.                     try {  
  79.                         URL url = new URL(params[0]);  
  80.                         HttpURLConnection urlConnection = (HttpURLConnection) url  
  81.                                 .openConnection();  
  82.                         urlConnection.connect();  
  83.                         int MAX = urlConnection.getContentLength();  
  84.                         InputStream inputStream = urlConnection  
  85.                                 .getInputStream();  
  86.                         ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();  
  87.                         byte[] b = new byte[1024];  
  88.                         int len = 0;  
  89.                         while ((len = inputStream.read(b)) != -1) {  
  90.                             byteArrayOutputStream.write(b, 0, len);  
  91.                         }  
  92.                         bitmap = BitmapFactory.decodeByteArray(  
  93.                                 byteArrayOutputStream.toByteArray(), 0, MAX);  
  94.                         inputStream.close();  
  95.                     } catch (MalformedURLException e) {  
  96.                         e.printStackTrace();  
  97.                     } catch (IOException e) {  
  98.                         e.printStackTrace();  
  99.                     }  
  100.                     return bitmap;  
  101.                 }  
  102.   
  103.                 @Override  
  104.                 protected void onPostExecute(Bitmap result) {  
  105.                     super.onPostExecute(result);  
  106.                     image.setImageBitmap(result);  
  107.                 }  
  108.   
  109.             }  
  110.             AsyncTaskLoadImage asyncTaskLoadImage = new AsyncTaskLoadImage();  
  111.             asyncTaskLoadImage.execute(imageURL);  
  112.             return convertView;  
  113.         }  
  114.   
  115.     }  
  116.   
  117. }  
       在这里让ListView显示100张图片,如图:

                                                                                                  Android_AsyncTask详解及其应用(二 )_RejectedExecutionException问题_第1张图片

       出现了两个异常,一个是:

                                          Android_AsyncTask详解及其应用(二 )_RejectedExecutionException问题_第2张图片

       这个异常很常见,就是经常说的OOM,内存溢出了,原因也是因为代码里并没有做OOM问题的优化处理,比如使用缓存、分页加载、重用等等。再看一下第二个异常:

                                 Android_AsyncTask详解及其应用(二 )_RejectedExecutionException问题_第3张图片

            那这个错误是什么呢?其实原因是当运行的AsyncTask 实例数量过多的时候会引发RejectedExecutionException异常。例如网络情况较差时,有大量的线程在等待。那如何解决这个问题呢?

            思路:

                     既然抛出了RejectedExecutionException这个异常,那就从它开始。翻看java的在线文档RejectedExecutionException,我们看到了这个错误的解释:

[java]  view plain copy
  1. public class RejectedExecutionException  
  2. extends RuntimeException  
  3. Exception thrown by an Executor when a task cannot be accepted for execution.                                                                             Since:  
  4. 1.5  
  5. See Also:  

                                                                  Android_AsyncTask详解及其应用(二 )_RejectedExecutionException问题_第4张图片

                   Serialized Form这个表明是Executor这个类不接受执行task时抛出的异常。而Executor是java1.5之后增加的java.util.concurrent包中操作多线程的主要类。可以执行多个RunnableTask,看来google的AsyncTask就是用的这套API。了解到这点,我们就可以打开AsyncTask的源码查看它到底是怎么实现的:打开源码之后,找到execute方法:

[java]  view plain copy
  1. 1     /** 
  2. 2      * Executes the task with the specified parameters. The task returns 
  3. 3      * itself (this) so that the caller can keep a reference to it. 
  4. 4      * 
  5. 5      * This method must be invoked on the UI thread. 
  6. 6      * 
  7. 7      * @param params The parameters of the task. 
  8. 8      * 
  9. 9      * @return This instance of AsyncTask. 
  10. 10      * 
  11. 11      * @throws IllegalStateException If {@link #getStatus()} returns either 
  12. 12      *         {@link AsyncTask.Status#RUNNING} or {@link AsyncTask.Status#FINISHED}. 
  13. 13      */  
  14. 14     public final AsyncTask<Params, Progress, Result> execute(Params... params) {  
  15. 15         if (mStatus != Status.PENDING) {  
  16. 16             switch (mStatus) {  
  17. 17                 case RUNNING:  
  18. 18                     throw new IllegalStateException("Cannot execute task:"  
  19. 19                             + " the task is already running.");  
  20. 20                 case FINISHED:  
  21. 21                     throw new IllegalStateException("Cannot execute task:"  
  22. 22                             + " the task has already been executed "  
  23. 23                             + "(a task can be executed only once)");  
  24. 24             }  
  25. 25         }  
  26. 26  
  27. 27         mStatus = Status.RUNNING;  
  28. 28  
  29. 29         onPreExecute();  
  30. 30  
  31. 31         mWorker.mParams = params;  
  32. 32         sExecutor.execute(mFuture);  
  33. 33  
  34. 34         return this;  
  35. 35     } 找到这里使用的Executor是sExecutor这个属性,这样来到它赋值的地方  
[java]  view plain copy
  1. 1     private static final String LOG_TAG = "AsyncTask";  
  2. 2  
  3. 3     private static final int CORE_POOL_SIZE = 5;  
  4. 4     private static final int MAXIMUM_POOL_SIZE = 128;  
  5. 5     private static final int KEEP_ALIVE = 10;  
  6. 6  
  7. 7     private static final BlockingQueue<Runnable> sWorkQueue =  
  8. 8             new LinkedBlockingQueue<Runnable>(10);  
  9. 9  
  10. 10     private static final ThreadFactory sThreadFactory = new ThreadFactory() {  
  11. 11         private final AtomicInteger mCount = new AtomicInteger(1);  
  12. 12  
  13. 13         public Thread newThread(Runnable r) {  
  14. 14             return new Thread(r, "AsyncTask #" + mCount.getAndIncrement());  
  15. 15         }  
  16. 16     };  
  17. 17  
  18. 18     private static final ThreadPoolExecutor sExecutor = new ThreadPoolExecutor(CORE_POOL_SIZE,  
  19. 19             MAXIMUM_POOL_SIZE, KEEP_ALIVE, TimeUnit.SECONDS, sWorkQueue, sThreadFactory);可以看到他是构造了一个ThreadPoolExecutor常量,保证new出多个AsyncTask都是使用这一个Executor。异常应该是它抛出的,我们看下这个类的文档,其中有一段是这样描述的:  
           Rejected tasks
 New tasks submitted in method execute(Runnable) will be rejected when the Executor has been shut down, and also when the Executor uses finite bounds for both maximum threads and work queue capacity, and is saturated. In either case, the execute method invokes the rejectedExecution(Runnable, ThreadPoolExecutor) method of its         RejectedExecutionHandler. Four predefined handler policies are provided:
  1.In the default ThreadPoolExecutor.AbortPolicy, the handler throws a runtime RejectedExecutionException upon rejection.
  2.In ThreadPoolExecutor.CallerRunsPolicy, the thread that invokes execute itself runs the task. This provides a simple feedback control mechanism that will slow    down the rate that new tasks are submitted.
  3.In ThreadPoolExecutor.DiscardPolicy, a task that cannot be executed is simply dropped.
  4.In ThreadPoolExecutor.DiscardOldestPolicy, if the executor is not shut down, the task at the head of the work queue is dropped, and then execution is retried     (which can fail again, causing this to be repeated.)
It is possible to define and use other kinds of RejectedExecutionHandler classes. Doing so requires some care especially when policies are designed to work only under particular capacity or queuing policies. 原来在Executor的队列和容量都达到最大时,再次增加新的task的时候将会进行拒绝行为,而默认的拒绝行为就是抛出这个RejectedExecutionException异常。       

                 那解决方案也就诞生了,就是复制一份AsyncTask的源码,自己重写这个初始化方法,增加相应的拒绝策略,后面就有几个可供选择的策略。修改AsyncTask源码如下

[java]  view plain copy
  1. 1     private static final ThreadPoolExecutor sExecutor = new ThreadPoolExecutor(CORE_POOL_SIZE,  
  2. 2     MAXIMUM_POOL_SIZE, KEEP_ALIVE, TimeUnit.SECONDS, sWorkQueue, sThreadFactory, new ThreadPoolExecutor.DiscardOldestPolicy());再次运行,OK,不会再抛出那个异常了。  
                 

   推荐博客:  1、Android AsyncTask behavior changes you should know

                       2、Android实战技巧:深入解析AsyncTask

                       3、AsyncTask 使用须知

                       4、Android多线程任务优化1:探讨AsyncTask的缺陷

                       5、解决java.util.concurrent.RejectedExecutionException问题


你可能感兴趣的:(Android_AsyncTask详解及其应用(二 )_RejectedExecutionException问题)