网站导航免费论文 原创论文 论文搜索 定作论文 网学软件 学术大家 资料中心 会员中心 问题解答 定作论文 大学论文导航 设计下载 最新论文 下载排行 原创论文 论文源代码
返回网学首页
网学联系
最新论文 推荐专题 热门论文 素材专题
当前位置: 网学 > 网学软件 > 网学开源 > 正文

AsyncTask的缺陷

来源:http://myeducs.cn 联系QQ:点击这里给我发消息 作者: 佚名 来源: 网络 发布时间: 14/08/06

【编者按】网学网网学开源频道为大家收集整理了“AsyncTask的缺陷 “提供大家参考,希望对大家有所帮助!

程序员必上的开发者服务平台 —— DevStore

导语:在开发Android应用的过程中,我们需要时刻注意保障应用的稳定性和界面响应性,因为不稳定或者响应速度慢的应用将会给用户带来非常差的交互体验。在越来越讲究用户体验的大环境下,用户也许会因为应用的一次Force Close(简称FC)或者延迟严重的动画效果而卸载你的应用。由于现在的应用大多需要异步连接网络,本系列文章就以构建网络应用为例,从稳定性和响应性两个角度分析多线程网络任务的性能优化方法。

概述:为了不阻塞UI线程(亦称主线程),提高应用的响应性,我们经常会使用新开线程的方式,异步处理那些导致阻塞的任务。

AsyncTask是Android为我们提供的方便编写异步任务的工具类,但是,在了解AsyncTask的实现原理之后,发现AsyncTask并不能满足我们所有的需求,使用不当还有可能导致应用FC。

本文主要通过分析AsyncTask提交任务的策略和一个具体的例子,说明AsyncTask的不足之处,至于解决办法,我们将在下篇再讲解。

分析

AsyncTask类包含一个全局静态的线程池,线程池的配置参数如下:

private static final int CORE_POOL_SIZE =5;//5个核心工作线程private static final int MAXIMUM_POOL_SIZE = 128;//最多128个工作线程private static final int KEEP_ALIVE = 1;//空闲线程的超时时间为1秒 private static final BlockingQueue<Runnable> sWorkQueue =           new LinkedBlockingQueue<Runnable>(10);//等待队列 private static final ThreadPoolExecutorsExecutor = new ThreadPoolExecutor(CORE_POOL_SIZE,           MAXIMUM_POOL_SIZE, KEEP_ALIVE, TimeUnit.SECONDS, sWorkQueue,sThreadFactory);//线程池是静态变量,所有的异步任务都会放到这个线程池的工作线程内执行。

我们这里不详细讲解ThreadPoolExecutor的原理,但将会讲解一个异步任务提交到AsyncTask的线程池时可能会出现的4种情况,并会提出在Android硬件配置普遍较低这个客观条件下,每个情况可能会出现的问题。

1、线程池中的工作线程少于5个时,将会创建新的工作线程执行异步任务(红色表示新任务,下同)


2、线程池中已经有5个线程,缓冲队列未满,异步任务将会放到缓冲队列中等待


3、线程池中已经有5个线程,缓冲队列已满,那么线程池将新开工作线程执行异步任务


问题:Android的设备一般不超过2个cpu核心,过多的线程会造成线程间切换频繁,消耗系统资源。

4、线程池中已经有128个线程,缓冲队列已满,如果此时向线程提交任务,将会抛出RejectedExecutionException


问题:抛出的错误不catch的话会导致程序FC。

好吧,理论分析之后还是要结合实际例子,我们通过实现一个模拟异步获取网络图片的例子,看看会不会出现上面提到的问题。

例子:使用GridView模拟异步加载大量图片

ActivityA.java

package com.zhuozhuo;import java.util.ArrayList;import java.util.Collection;import java.util.HashMap;import java.util.Iterator;import java.util.List;import java.util.ListIterator;import java.util.Map;import android.app.Activity;import android.app.AlertDialog;import android.app.Dialog;import android.app.ListActivity;import android.app.ProgressDialog;import android.content.Context;import android.content.DialogInterface;import android.content.Intent;import android.database.Cursor;import android.graphics.Bitmap;import android.os.AsyncTask;import android.os.Bundle;import android.provider.ContactsContract;import android.util.Log;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.widget.AbsListView;import android.widget.AbsListView.OnScrollListener;import android.widget.Adapter;import android.widget.AdapterView;import android.widget.AdapterView.OnItemClickListener;import android.widget.BaseAdapter;import android.widget.GridView;import android.widget.ImageView;import android.widget.ListAdapter;import android.widget.SimpleAdapter;import android.widget.TextView;import android.widget.Toast;public class ActivityA extends Activity {        private GridView mGridView;private List<HashMap<String, Object>> mData;private BaseAdapter mAdapter;private ProgressDialog mProgressDialog;private static final int DIALOG_PROGRESS = 0;@Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.main);        mGridView = (GridView) findViewById(R.id.gridview);        mData = new ArrayList<HashMap<String,Object>>();        mAdapter = new CustomAdapter();                       mGridView.setAdapter(mAdapter);    }protected void onStart () {super.onStart();new GetGridDataTask().execute(null);//执行获取数据的任务}    @Override    protected Dialog onCreateDialog(int id) {        switch (id) {        case DIALOG_PROGRESS:            mProgressDialog = new ProgressDialog(ActivityA.this);            mProgressDialog.setMessage("正在获取数据");            mProgressDialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);            return mProgressDialog;               }        return null;}class CustomAdapter extends BaseAdapter {CustomAdapter() {}@Overridepublic int getCount() {return mData.size();}@Overridepublic Object getItem(int position) {return mData.get(position);}@Overridepublic long getItemId(int position) {return 0;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {View view = convertView;ViewHolder vh;if(view == null) {view = LayoutInflater.from(ActivityA.this).inflate(R.layout.list_item, null);vh = new ViewHolder();vh.tv = (TextView) view.findViewById(R.id.textView);vh.iv = (ImageView) view.findViewById(R.id.imageView);view.setTag(vh);}vh = (ViewHolder) view.getTag();vh.tv.setText((String) mData.get(position).get("title"));Integer id = (Integer) mData.get(position).get("pic");if(id != null) {vh.iv.setImageResource(id);}else {vh.iv.setImageBitmap(null);}FifoAsyncTask task = (FifoAsyncTask) mData.get(position).get("task");if(task == null || task.isCancelled()) {Log.d("Test", "" + position);mData.get(position).put("task", new GetItemImageTask(position).execute(null));//执行获取图片的任务}return view;}}static class ViewHolder {TextView tv;ImageView iv;}class GetGridDataTask extends FifoAsyncTask<Void, Void, Void> {protected void onPreExecute () {mData.clear();mAdapter.notifyDataSetChanged();showDialog(DIALOG_PROGRESS);//打开等待对话框}@Overrideprotected Void doInBackground(Void... params) {try {Thread.sleep(500);//模拟耗时的网络操作} catch (InterruptedException e) {e.printStackTrace();}for(int i = 0; i < 200; i++) {HashMap<String, Object> hm = new HashMap<String, Object>();hm.put("title", "Title");mData.add(hm);}return null;}protected void onPostExecute (Void result) {mAdapter.notifyDataSetChanged();//通知ui界面更新dismissDialog(DIALOG_PROGRESS);//关闭等待对话框}}class GetItemImageTask extends FifoAsyncTask<Void, Void, Void> {int pos;GetItemImageTask(int pos) {this.pos = pos;}@Overrideprotected Void doInBackground(Void... params) {try {Thread.sleep(2000); //模拟耗时的网络操作} catch (InterruptedException e) {e.printStackTrace();}mData.get(pos).put("pic", R.drawable.icon);return null;}protected void onPostExecute (Void result) {mAdapter.notifyDataSetChanged();//通知ui界面更新}}}

由运行图可见

当网络情况较差,异步任务不能尽快完成执行的情况下,新开的线程会造成listview滑动不流畅。当开启的工作线程过多时,还有出现FC的可能。

至此,你还相信万能的AsyncTask吗?至于你信不信,反正我不信。

总结:

AsyncTask可能存在新开大量线程消耗系统资源和导致应用FC的风险,因此,我们需要根据自己的需求自定义不同的线程池,由于篇幅问题,将留到下篇再讲。

文章转自:http://blog.csdn.net/mylzc/article/details/6784415

网学推荐

免费论文

原创论文

设为首页 | 加入收藏 | 论文首页 | 论文专题 | 设计下载 | 网学软件 | 论文模板 | 论文资源 | 程序设计 | 关于网学 | 站内搜索 | 网学留言 | 友情链接 | 资料中心
版权所有 QQ:3710167 邮箱:3710167@qq.com 网学网 [Myeducs.cn] 您电脑的分辨率是 像素
Copyright 2008-2015 myeducs.Cn www.myeducs.Cn All Rights Reserved 湘ICP备09003080号