- 浏览: 269863 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
androidios2014:
运行有滚没有把分组选项置顶,楼主帮忙给个源码学习7196628 ...
【so easy~】完全仿QQ好友列表,自定义ExpandableListView! -
hnu_hgc:
求代码,3243544018@qq.com
Android释放内存(结束进程) -
hnu_hgc:
求代码工程
Android释放内存(结束进程) -
l657757077:
楼主给个源码学习学习。邮箱:657757077@qq.com
【so easy~】完全仿QQ好友列表,自定义ExpandableListView! -
hao_yh:
楼主,修改后的代码能发我一份吗?感谢。感觉现在是只要界面创建就 ...
【so easy~】完全仿QQ好友列表,自定义ExpandableListView!
导语:在开发Android应用的过程中,我们需要时刻注意保障应用的稳定性和界面响应性,因为不稳定或者响应速度慢的应用将会给用户带来非常差的交互体验。在越来越讲究用户体验的大环境下,用户也许会因为应用的一次Force Close(简称FC)或者延迟严重的动画效果而卸载你的应用。由于现在的应用大多需要异步连接网络,本系列文章就以构建网络应用为例,从稳定性和响应性两个角度分析多线程网络任务的性能优化方法。
概述:为了不阻塞UI线程(亦称主线程),提高应用的响应性,我们经常会使用新开线程的方式,异步处理那些导致阻塞的任务(如要了解Android异步处理的实现方式和原理,请先阅读《Android异步处理系列文章索引》)。
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() {
- }
- @Override
- public int getCount() {
- return mData.size();
- }
- @Override
- public Object getItem(int position) {
- return mData.get(position);
- }
- @Override
- public long getItemId(int position) {
- return 0;
- }
- @Override
- public 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);//打开等待对话框
- }
- @Override
- protected 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;
- }
- @Override
- protected 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的风险,因此,我们需要根据自己的需求自定义不同的线程池,由于篇幅问题,将留到下篇再讲。
发表评论
-
【so easy】Android手动打包流程,完全详细版!
2012-08-09 12:19 5224由于是先写的word文档,所以排版不好,看不清的朋友可以直 ... -
转:Bitmap造成OOM的讨论与解决方案
2012-07-26 16:08 11227android BitmapFactory的OutOfM ... -
【so easy~】自定义上下menu菜单
2012-06-25 18:14 2410最近做了自定义菜单,许多阅读器的样子,都是点击屏幕中央弹 ... -
【so easy】Tabhost每个tab中放入多个Activity!
2012-04-25 18:14 5103最近做完了一个客户端项目 外面是个tabhost(用于底部t ... -
【so easy~】完全退出
2012-03-08 16:52 1730String packagename = getPac ... -
【BUG解决】WebView cookies清理
2012-03-05 15:21 12091今天在项目中发现一个BUG 在使用新浪微博账户登录应用时,w ... -
Project has no default.properties file! Edit the project properties to set one.
2012-02-20 14:02 1867导入新项目 有时会:Project has no def ... -
【so easy~】完全仿QQ好友列表,自定义ExpandableListView!
2012-01-10 16:14 25066最近,需要做一个可展开的listview,不禁想起了Expan ... -
Android释放内存(结束进程)
2011-12-30 15:45 10199这两天在搞个内存清理小插件,网上很少这方面资料,之前找过清 ... -
【so easy~】 Gallery 模仿Flash广告栏~!附源码
2011-12-19 16:24 5644先上个效果图~ http://androiddada.i ... -
android 获得手机信息
2011-12-02 14:22 1370Android中与电话功能相关的类是 Telephony ... -
【so easy~】 Webview离线功能(优先cache缓存+cache缓存管理)!
2011-11-28 17:05 39700在做Webview显示服务器的html功能时 需要加入离线功能 ... -
转!SQL:char、varchar、text和nchar、nvarchar、ntext的区别
2011-11-21 16:07 1767SQL:char、varchar、text和nchar、n ... -
【so easy~】 底部菜单可移动焦点~!(仿网易新闻等应用)
2011-11-18 10:51 3641最近比较懒惰,也没有更新博客。今天就把刚刚实现的一个小效果分享 ... -
转:shape 渐变!描边!圆角!漂亮的自定义背景
2011-11-09 09:47 3957在实现背景渐变的帖子里 http://androi ... -
ListView item渐变背景(ListView必备!)
2011-11-07 15:32 7168在看他人的代码学习时,无意中发现的。很多应用中的ListVi ... -
Android之系统自带的文字外观设置及实际显示效果图
2011-11-07 14:50 1309Android之系统自带的文字外观设置及实际显示效果图 ... -
【so easy~】 Intent实现 <分享>功能!~
2011-10-31 15:06 1531http://androiddada.iteye.com/ ... -
【so easy~】 GPS的那点儿事儿~!
2011-10-28 14:27 2042http://androiddada.iteye.com/ ... -
Service使用总结 与sdk部分翻译
2011-10-27 15:12 17111. Service SDK翻译 ...
相关推荐
这是Pro Android学习系列中AsyncTask部分的例子源代码。相关学习笔记见:http://blog.csdn.net/flowingflying/article/details/6212512
4. 一个AsyncTask实例只能执行一次,重复执行会导致异常。 以下是一个简单的示例,展示了如何使用AsyncTask从网络获取网页源代码并在TextView中显示: ```java public class MainActivity extends Activity { ...
Android异步处理二:使用AsyncTask异步更新UI界面。
compile ' com.label305:asynctask:x.x.x. ' SimpleAsyncTask 在执行不会引发检查的Exception工作时使用SimpleAsyncTask : public class MyAsyncTask extends SimpleAsyncTask< String> { @Override protected...
AsyncTask 是 Android 平台上一个非常重要的工具类,用于在后台线程执行耗时操作,然后在 UI 线程更新结果,从而避免了因长时间运行任务而导致的 ANR(Application Not Responding)错误。在深入探讨 AsyncTask 的...
在Android应用开发中,异步任务(AsyncTask)是一种常用的方法,用于在后台线程执行耗时操作,如网络请求、文件下载等,避免阻塞主线程,提高用户体验。本项目"AndroidAsyncTaskDownLoadImage"专注于利用AsyncTask来...
在这个`AsyncTask`作为带有接口的类的示例中,我们将深入探讨`AsyncTask`的工作原理,以及如何结合接口来实现更灵活的通信。 `AsyncTask`类是一个抽象模板类,它提供了三个泛型参数:`Params`、`Progress`和`Result...
因为`AsyncTask`默认是作为静态内部类存在的,如果Activity提前销毁但`AsyncTask`仍在运行,就可能导致Activity无法正常回收,引发内存泄漏。为了解决这个问题,可以考虑将`AsyncTask`作为成员变量,或者使用弱引用...
在Android开发中,`AsyncTask`和API的使用是至关重要的技术点,它们涉及到应用程序的异步处理和网络数据交互。让我们深入探讨这两个概念及其在实际应用中的运用。 `AsyncTask`是Android SDK提供的一种轻量级的异步...
`AsyncTask`是Android SDK提供的一种轻量级的并发工具,用于处理短时间的、与UI交互的任务。然而,原生的`AsyncTask`并不支持链式调用,这使得代码组织和扩展性有所限制。`SimpleAsyncTask`就是为了解决这个问题而...
如果在Activity中创建了一个AsyncTask并在后台执行,当Activity被销毁时,AsyncTask可能会继续执行,导致内存泄漏。因此,需要在Activity的`onPause()`或`onDestroy()`中取消任务。 2. **线程限制**:Android系统...
首先,Android系统为了解决长时间运行的任务在主线程执行导致的UI冻结问题,提供了AsyncTask。AsyncTask分为三个基本步骤:onPreExecute()、doInBackground()和onPostExecute()。 1. **onPreExecute()**:这是在...
如果不妥善处理,可能会导致内存泄漏或意外行为。因此,通常建议在Activity的onDestroy()方法中取消正在执行的AsyncTask。 3. **版本兼容性**:自Android 3.0(API级别11)开始,AsyncTask默认在单独的后台线程上...
- **生命周期管理**:由于Activity或Fragment的生命周期,若在这些组件中使用AsyncTask,需要妥善处理配置变更(如屏幕旋转)可能导致的任务丢失问题。 - **线程限制**:默认情况下,AsyncTask内部维护了一个线程池...
- AsyncTask的数量有限制,大量并发可能导致问题,因此建议谨慎使用,特别是大量短时间的任务。 - 不要在Activity的onCreate()方法中直接启动AsyncTask,因为Activity可能还没完全创建,可能导致异常。 - 当...
最近项目中要做一个带进度条的上传文件的功能,学习了AsyncTask,使用起来比较方便,将几个方法实现就行,另外做了一个很简单的demo,希望能对大家有帮助,在程序中设好文件路径和服务器IP即可。 demo运行截图: ...
- 由于Android系统的内存管理机制,Activity销毁后,与之关联的AsyncTask可能会被系统回收,导致异常。因此,若Activity可能被销毁,应在`onDestroy()`中取消AsyncTask的执行。 - Android 3.0及以上版本,...