转:android listview局部刷新和模拟应用下载
在 android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用 notifyDataSetChanged()这个函数,但是它会更新listview中所有可视范围内的item,这样对性能肯定会有影响。比较常见的 情景是android应用商店中的下载列表,当我们下载一款游戏的时候,只需要更新这款游戏对应的进度就可以了。本文就来模拟android应用商店的游 戏下载,实现对listview的局部刷新,只实现一个简单的demo,不去真的下载文件。
1. 首先来创建代表应用商店中的app文件的类:AppFile.java,包含了一些基本的属性,源码:
- package com.alexzhou.downloadfile;
- /**
- * author:alexzhou
- * email :zhoujiangbohai@163.com
- * date :2013-1-27
- *
- * 游戏列表中的app文件
- **/
- public class AppFile {
- public int id;
- public String name;
- // app的大小
- public int size;
- // 已下载大小
- public int downloadSize;
- // 下载状态:正常,正在下载,暂停,等待,已下载
- public int downloadState;
- }
2.
由于实际开发时,AppFile的属性比较多,这里创建一个辅助类:DownloadFile.java,代表下载中的文件,源码:
- package com.alexzhou.downloadfile;
- /**
- * author:alexzhou
- * email :zhoujiangbohai@163.com
- * date :2013-1-27
- *
- * 下载的文件
- **/
- public class DownloadFile {
- public int downloadID;
- public int downloadSize;
- public int totalSize;
- public int downloadState;
- }
3.
接下来需要一个下载管理类:DownloadManager.java,它管理所有下载任务。当同时下载很多任务的时候,界面会卡,所以指定只能同时下载
3个任务,每个任务会启动一个线程,这里使用了ExecutorService线程池。当提交了超过三个下载任务时,只执行前3个任务,第四个任务会等到
前面有一个下载完成后再下载,以此类推。这里还用到了android提供的一个工具类SparseArray,它是用来替代HashMap的,性能比
HashMap要好。下面看源码:
- package com.alexzhou.downloadfile;
- import java.util.ArrayList;
- import java.util.concurrent.ExecutorService;
- import java.util.concurrent.Executors;
- import android.os.Handler;
- import android.os.Message;
- import android.util.Log;
- import android.util.SparseArray;
- /**
- author:alexzhou
- email :zhoujiangbohai@163.com
- date :2013-1-27
- 下载管理
- **/
- public class DownloadManager {
- // 下载状态:正常,暂停,下载中,已下载,排队中
- public static final int DOWNLOAD_STATE_NORMAL = 0x00;
- public static final int DOWNLOAD_STATE_PAUSE = 0x01;
- public static final int DOWNLOAD_STATE_DOWNLOADING = 0x02;
- public static final int DOWNLOAD_STATE_FINISH = 0x03;
- public static final int DOWNLOAD_STATE_WAITING = 0x04;
- // SparseArray是android中替代Hashmap的类,可以提高效率
- private SparseArray<DownloadFile> downloadFiles = new SparseArray<DownloadFile>();
- // 用来管理所有下载任务
- private ArrayList<DownloadTask> taskList = new ArrayList<DownloadTask>();
- private Handler mHandler;
- private final static Object syncObj = new Object();
- private static DownloadManager instance;
- private ExecutorService executorService;
- private DownloadManager()
- {
- // 最多只能同时下载3个任务,其余的任务排队等待
- executorService = Executors.newFixedThreadPool(3);
- }
- public static DownloadManager getInstance()
- {
- if(null == instance)
- {
- synchronized(syncObj) {
- instance = new DownloadManager();
- }
- return instance;
- }
- return instance;
- }
- public void setHandler(Handler handler) {
- this.mHandler = handler;
- }
- // 开始下载,创建一个下载线程
- public void startDownload(DownloadFile file) {
- downloadFiles.put(file.downloadID, file);
- DownloadTask task = new DownloadTask(file.downloadID);
- taskList.add(task);
- executorService.submit(task);
- }
- public void stopAllDownloadTask() {
- while(taskList.size() != 0)
- {
- DownloadTask task = taskList.remove(0);
- // 可以在这里做其他的处理
- task.stopTask();
- }
- // 会停止正在进行的任务和拒绝接受新的任务
- executorService.shutdownNow();
- }
- // 下载任务
- class DownloadTask implements Runnable {
- private boolean isWorking = false;
- private int downloadId;
- public DownloadTask(int id)
- {
- this.isWorking = true;
- this.downloadId = id;
- }
- public void stopTask()
- {
- this.isWorking = false;
- }
- // 更新listview中对应的item
- public void update(DownloadFile downloadFile)
- {
- Message msg = mHandler.obtainMessage();
- if(downloadFile.totalSize == downloadFile.downloadSize)
- downloadFile.downloadState = DOWNLOAD_STATE_FINISH;
- msg.obj = downloadFile;
- msg.sendToTarget();
- }
- public void run() {
- // 更新下载文件的状态
- DownloadFile downloadFile = downloadFiles.get(downloadId);
- downloadFile.downloadState = DOWNLOAD_STATE_DOWNLOADING;
- while(isWorking)
- {
- // 检测是否下载完成
- if(downloadFile.downloadState != DOWNLOAD_STATE_DOWNLOADING)
- {
- downloadFiles.remove(downloadFile.downloadID);
- taskList.remove(this);
- isWorking = false;
- break;
- }
- //Log.e("", "downloadSize="+downloadFile.downloadSize+"; size="+downloadFile.totalSize);
- // 这里只是模拟了下载,每一秒更新一次item的下载状态
- if(downloadFile.downloadSize <= downloadFile.totalSize)
- {
- this.update(downloadFile);
- }
- if(downloadFile.downloadSize < downloadFile.totalSize)
- {
- try {
- Thread.sleep(100);
- } catch (InterruptedException e) {
- e.printStackTrace();
- downloadFile.downloadState = DOWNLOAD_STATE_PAUSE;
- this.update(downloadFile);
- downloadFiles.remove(downloadId);
- isWorking = false;
- break;
- }
- ++ downloadFile.downloadSize;
- }
- }
- }
- }
- }
4. 接下来就需要实现listview的adapter了,这里比较重要的一个函数是updateView,这是实现listview局部刷新的关键,通过索引index得到listview中对应位置的子view,然后再更新该view的数据。源码:
- package com.alexzhou.downloadfile;
- import android.content.Context;
- import android.graphics.drawable.Drawable;
- import android.os.Handler;
- import android.os.Message;
- import android.util.Log;
- import android.util.SparseArray;
- import android.view.LayoutInflater;
- import android.view.View;
- import android.view.View.OnClickListener;
- import android.view.ViewGroup;
- import android.widget.BaseAdapter;
- import android.widget.Button;
- import android.widget.ImageView;
- import android.widget.LinearLayout;
- import android.widget.ListView;
- import android.widget.TextView;
- /**
- author:alexzhou
- email :zhoujiangbohai@163.com
- date :2013-1-27
- app列表的数据适配器
- **/
- public class AppListAdapter extends BaseAdapter {
- private SparseArray<AppFile> dataList = null;
- private LayoutInflater inflater = null;
- private Context mContext;
- private DownloadManager downloadManager;
- private ListView listView;
- public AppListAdapter(Context context, SparseArray<AppFile> dataList) {
- this.inflater = (LayoutInflater) context
- .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
- this.dataList = dataList;
- this.mContext = context;
- this.downloadManager = DownloadManager.getInstance();
- this.downloadManager.setHandler(mHandler);
- }
- public void setListView(ListView view)
- {
- this.listView = view;
- }
- @Override
- public int getCount() {
- return dataList.size();
- }
- @Override
- public Object getItem(int position) {
- return dataList.get(position);
- }
- @Override
- public long getItemId(int position) {
- return position;
- }
- // 改变下载按钮的样式
- private void changeBtnStyle(Button btn, boolean enable)
- {
- if(enable)
- {
- btn.setBackgroundResource(R.drawable.btn_download_norm);
- }
- else
- {
- btn.setBackgroundResource(R.drawable.btn_download_disable);
- }
- btn.setEnabled(enable);
- }
- @Override
- public View getView(int position, View convertView, ViewGroup parent) {
- final ViewHolder holder;
- if (null == convertView) {
- holder = new ViewHolder();
- convertView = inflater.inflate(R.layout.listitem_app, null);
- holder.layout = (LinearLayout) convertView
- .findViewById(R.id.gamelist_item_layout);
- holder.icon = (ImageView) convertView
- .findViewById(R.id.app_icon);
- holder.name = (TextView) convertView
- .findViewById(R.id.app_name);
- holder.size = (TextView) convertView
- .findViewById(R.id.app_size);
- holder.btn = (Button) convertView
- .findViewById(R.id.download_btn);
- convertView.setTag(holder);
- } else {
- holder = (ViewHolder) convertView.getTag();
- }
- // 这里position和app.id的值是相等的
- final AppFile app = dataList.get(position);
- //Log.e("", "id="+app.id+", name="+app.name);
- holder.name.setText(app.name);
- holder.size.setText((app.downloadSize * 100.0f / app.size) + "%");
- Drawable drawable = mContext.getResources().getDrawable(R.drawable.app_icon);
- holder.icon.setImageDrawable(drawable);
- switch(app.downloadState)
- {
- case DownloadManager.DOWNLOAD_STATE_NORMAL:
- holder.btn.setText("下载");
- this.changeBtnStyle(holder.btn, true);
- break;
- case DownloadManager.DOWNLOAD_STATE_DOWNLOADING:
- holder.btn.setText("下载中");
- this.changeBtnStyle(holder.btn, false);
- break;
- case DownloadManager.DOWNLOAD_STATE_FINISH:
- holder.btn.setText("已下载");
- this.changeBtnStyle(holder.btn, false);
- break;
- case DownloadManager.DOWNLOAD_STATE_WAITING:
- holder.btn.setText("排队中");
- this.changeBtnStyle(holder.btn, false);
- break;
- }
- holder.btn.setOnClickListener(new OnClickListener() {
- @Override
- public void onClick(View v) {
- DownloadFile downloadFile = new DownloadFile();
- downloadFile.downloadID = app.id;
- downloadFile.downloadState = DownloadManager.DOWNLOAD_STATE_WAITING;
- app.downloadState = DownloadManager.DOWNLOAD_STATE_WAITING;
- downloadFile.downloadSize = app.downloadSize;
- downloadFile.totalSize = app.size;
- holder.btn.setText("排队中");
- changeBtnStyle(holder.btn, false);
- downloadManager.startDownload(downloadFile);
- }
- });
- return convertView;
- }
- static class ViewHolder {
- LinearLayout layout;
- ImageView icon;
- TextView name;
- TextView size;
- Button btn;
- }
- private Handler mHandler = new Handler() {
- public void handleMessage(Message msg)
- {
- DownloadFile downloadFile = (DownloadFile)msg.obj;
- AppFile appFile = dataList.get(downloadFile.downloadID);
- appFile.downloadSize = downloadFile.downloadSize;
- appFile.downloadState = downloadFile.downloadState;
- // notifyDataSetChanged会执行getView函数,更新所有可视item的数据
- //notifyDataSetChanged();
- // 只更新指定item的数据,提高了性能
- updateView(appFile.id);
- }
- };
- // 更新指定item的数据
- private void updateView(int index)
- {
- int visiblePos = listView.getFirstVisiblePosition();
- int offset = index - visiblePos;
- //Log.e("", "index="+index+"visiblePos="+visiblePos+"offset="+offset);
- // 只有在可见区域才更新
- if(offset < 0) return;
- View view = listView.getChildAt(offset);
- final AppFile app = dataList.get(index);
- ViewHolder holder = (ViewHolder)view.getTag();
- //Log.e("", "id="+app.id+", name="+app.name);
- holder.name.setText(app.name);
- holder.size.setText((app.downloadSize * 100.0f / app.size) + "%");
- Drawable drawable = mContext.getResources().getDrawable(R.drawable.app_icon);
- holder.icon.setImageDrawable(drawable);
- switch(app.downloadState)
- {
- case DownloadManager.DOWNLOAD_STATE_DOWNLOADING:
- holder.btn.setText("下载中");
- this.changeBtnStyle(holder.btn, false);
- break;
- case DownloadManager.DOWNLOAD_STATE_FINISH:
- holder.btn.setText("已下载");
- this.changeBtnStyle(holder.btn, false);
- break;
- }
- }
- }
布局文件listitem_app.xml:
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:id="@+id/gamelist_item_layout"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:gravity="center_vertical"
- android:background="@drawable/style_listitem_background"
- android:paddingBottom="5dp"
- android:paddingTop="5dp" >
- <ImageView
- android:id="@+id/app_icon"
- android:layout_width="53dip"
- android:layout_height="53dip"
- android:layout_marginLeft="5dip"
- android:adjustViewBounds="false"
- android:padding="5dp" />
- <LinearLayout
- android:layout_width="match_parent"
- android:layout_height="60dp"
- android:layout_marginLeft="5dp"
- android:layout_weight="1"
- android:gravity="center_vertical"
- android:orientation="vertical" >
- <TextView
- android:id="@+id/app_name"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:singleLine="true"
- android:text=""
- android:textColor="#000000"
- android:textSize="13sp" />
- <TextView
- android:id="@+id/app_size"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textColor="#000000"
- android:textSize="10sp" />
- </LinearLayout>
- <Button
- android:id="@+id/download_btn"
- android:layout_width="55dip"
- android:layout_height="30dip"
- android:layout_marginRight="10dip"
- android:background="@drawable/style_btn_download"
- android:focusable="false"
- android:text="@string/download"
- android:textColor="#ffffffff"
- android:textSize="12sp" />
- </LinearLayout>
listview中item样式文件style_listitem_background.xml:
- <?xml version="1.0" encoding="utf-8"?>
- <selector xmlns:android="http://schemas.android.com/apk/res/android">
- <!-- 没有焦点时的背景颜色 -->
- <item android:state_window_focused="false" >
- <shape>
- <gradient
- android:startColor="#ffffff"
- android:endColor="#E3E3E3"
- android:angle="-90" />
- </shape>
- </item>
- <!-- 非触摸模式下获得焦点并单击时的背景颜色 -->
- <item android:state_focused="true" android:state_pressed="true"
- android:drawable="@drawable/bg_listview_item_selected" />
- <!--触摸模式下单击时的背景颜色 -->
- <item android:state_focused="false" android:state_pressed="true"
- android:drawable="@drawable/bg_listview_item_selected" />
- <!--选中时的背景颜色 -->
- <item android:state_selected="true" android:drawable="@drawable/bg_listview_item_selected" />
- <!--获得焦点时的背景 颜色-->
- <item android:state_focused="true" android:drawable="@drawable/bg_listview_item_selected" />
- </selector>
item中的button样式文件style_btn_download.xml:
- <?xml version="1.0" encoding="utf-8"?>
- <selector xmlns:android="http://schemas.android.com/apk/res/android">
- <item android:state_pressed="true"
- android:drawable="@drawable/btn_download_pressed" />
- <item android:drawable="@drawable/btn_download_norm" />
- </selector>
字符文件strings.xml:
- <?xml version="1.0" encoding="utf-8"?>
- <resources>
- <string name="app_name">AndroidDownloadFile</string>
- <string name="download">下载</string>
- </resources>
5. 最后创建MainActivity.java,源码:
- package com.alexzhou.downloadfile;
- import android.app.Activity;
- import android.os.Bundle;
- import android.util.SparseArray;
- import android.widget.ListView;
- public class MainActivity extends Activity
- {
- private SparseArray<AppFile> appList = new SparseArray<AppFile>();
- private ListView listView;
- @Override
- public void onCreate(Bundle savedInstanceState)
- {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- initData();
- initUI();
- }
- private void initData()
- {
- for(int i =0; i<20; i++)
- {
- AppFile app = new AppFile();
- app.name = "快玩游戏--" + (i+1);
- app.size = 100;
- app.id = i;
- app.downloadState = DownloadManager.DOWNLOAD_STATE_NORMAL;
- app.downloadSize = 0;
- appList.put(app.id, app);
- }
- }
- private void initUI()
- {
- listView = (ListView)this.findViewById(R.id.listview);
- AppListAdapter adapter = new AppListAdapter(this, appList);
- adapter.setListView(listView);
- listView.setAdapter(adapter);
- }
- @Override
- protected void onDestroy() {
- super.onDestroy();
- DownloadManager.getInstance().stopAllDownloadTask();
- }
- }
布局文件activity_main.xml:
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:orientation="vertical"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- >
- <ListView
- android:id="@+id/listview"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- android:fastScrollEnabled="true"
- />
- </LinearLayout>
到此为止,代码部分已经全部完成了,下面来看看最终效果图:
这里对比一下分别使用updateView和notifyDataSetChanged时,有什么不一样,看看打印日志:
(1)使用notifyDataSetChanged时,listview可视范围内的所有子项都更新了。
(2)使用updateView时,只更新了指定的子项。
实例源码地址
: http://pan.baidu.com/share/link?shareid=229182&uk=167811495
转载请注明来自:Alex
Zhou的程序世界,本文链接:http://codingnow.cn/Android/1059.html
转:android listview局部刷新和模拟应用下载的更多相关文章
- Android listview局部刷新和模拟应用下载(zhu)
在android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用notifyDataSetChanged()这个函数,但是它会更新listview中所有可 ...
- Android 结合实际项目学会ListView局部刷新和相关知识《一》
转载本专栏博客,请注明出处:道龙的博客 最近在公司参与的项目中有一个界面需要做局部UI更新处理,把其化烦为简为Demoi形式写在这里.我们还是运行该Demo,知道ListView局部刷新的使用场景:( ...
- 再说Android RecyclerView局部刷新那个坑
RecyclerView局部刷新大家都遇到过,有时候还说会遇见图片闪烁的问题. 优化之前的效果: 优化之后的效果: 如果想单独更新一个item,我们通常会这样做,代码如下: mLRecyclerV ...
- Android RecyclerView局部刷新那个坑
关键:public final void notifyItemChanged(int position, Object payload) RecyclerView局部刷新大家都遇到过,有时候还说会遇见 ...
- Android ListView 单条刷新方法实践及原理解析
对于使用listView配合adapter进行刷新的方法大家都不陌生,先刷新adapter里的数据,然后调用notifydatasetchange通知listView刷新界面. 方法虽然简单,但这里面 ...
- jquerymobile listview 局部刷新
function onSuccess(data, status) { data = $.trim(data); // alert(data); // return; if (data) { $('#l ...
- android:ListView的局部刷新
1.简介 对于android中的ListView刷新机制,大多数的程序员都是很熟悉的,修改或者添加adapter中的数据源之后,然后调用notifyDataSetChanged()刷新ListView ...
- [置顶]
android ListView包含Checkbox滑动时状态改变
题外话: 在xamarin android的开发中基本上所有人都会遇到这个小小的坎,的确有点麻烦,当时我也折腾了好一半天,如果你能看到这篇博客,说明你和我当初也是一样的焦灼,如果你想解决掉这个小小的坎 ...
- android listview用adapter.notifyDataSetChanged()无法刷新每项的图标
http://blog.csdn.net/caizhegnhao/article/details/41318575 今天在开发中遇到一个很奇怪的listview的问题. 这个问题情景是我的应用需要做一 ...
随机推荐
- 图解HTTP总结(4)——返回结果的HTTP状态码
HTTP状态码负责表示客户端HTTP请求的返回结果.标记服务器端的处理是否正常.通知出现的错误等工作. 状态码的类别 2XX 成功 200 OK 表示从客户端发来的请求在服务器端被正常处理了. 在响应 ...
- POJ:3320-Jessica's Reading Problem(尺取法)
Jessica's Reading Problem Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 15224 Accepted: ...
- ABAP自定义截取字符串长度函数
SAP 中strlen()只能计算字符串的个数,不能计算含有中文字符串的长度,如字符串“SAP大波霸”,strlen('SAP大波霸') = 6,其实真实长度为3+3*2 = 9.我们可以通过cl_a ...
- Smail 中的一些点
smali中所有操作都需要经过寄存器, 本地寄存器以v开头, 参数寄存器以p开头, 非static方法中p0是this 没有-object后缀的操作指令表示操作的对象是基本类型 invoke-dire ...
- PHP.34-TP框架商城应用实例-后台10-商品分类-需求分析、创建无限级商品分类,递归
商品管理需求分析 1.实现商品无限级分类管理[类似京东三级分类] 2.添加商品时要指定商品属于一个主分类和多个扩展分类[扩展分类可以是其他主分类] 3.商品列表中可以根据分类搜索商品 a) 搜索一个分 ...
- IAR配置ICF到项目的实现方法
以STM8为例: 将项目使用的MCU型号的icf文件拷贝到项目里面,icf一般放在 C:\Program Files\IAR Systems\Embedded Workbench 6.5\stm8\c ...
- 3,Flask 中的模板语言 Jinja2 及 render_template 的深度用法
Flask中默认的模板语言是Jinja2 现在我们来一步一步的学习一下 Jinja2 捎带手把 render_template 中留下的疑问解决一下 首先我们要在后端定义几个字符串,用于传递到前端 S ...
- mysql安装与基本管理,mysql密码破解
一.MySQL介绍 MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,目前属于 Oracle 旗下公司.MySQL 最流行的关系型数据库管理系统,在 WEB 应用方面MySQL是 ...
- poj2001Shortest Prefixes(trie)
Shortest Prefixes Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 18687 Accepted: 808 ...
- 《Cracking the Coding Interview》——第18章:难题——题目1
2014-04-29 00:56 题目:不用算数运算,完成加法. 解法:那就位运算吧,用加法器的做法就可以了. 代码: // 18.1 add two numbers wihout using ari ...