缘由

我们在用ListView展现数据的时候。比如展现联系人,如果联系人太多就会出现卡的现象,比如如果有1000多条数据,从数据库里查询,然后装载到List容器这段时间是比较耗时的。虽然我们可以用asyncTask来单独开启一个子线程加载。一次查看那么多,未免有点多余。是否可以通过先装载30条数据,如果用户需要我们在继续查询并且展示后面的数据,这样一来可以提升。使用速率。

实现方法

1、数据库分页查询

首先写SqlHelper指定每次查询数据库多少数据。代码如下

/**
* 分页查询黑名单
*
* @param limit
* 限制数量
* @param offset
* 开始位置
* @return
*/
public List<BlackNumInfo> getPartBlackNum(int limit, int offset)
{
//查询返回的数据容器
List<BlackNumInfo> blackNums = null;
SQLiteDatabase db = dbHelper.getReadableDatabase();

String[] columns = new String[] { NUM, MODE, ID };
String selection = null;
String[] selectionArgs = null;
String groupBy = null;
String having = null;
String orderBy = null; // 排序功能
String limitsr = offset + "," + limit; // 分页数据

Cursor cursor = db.query(TABLE, columns, selection, selectionArgs, groupBy, having,
orderBy, limitsr);
//如果没有数据就返回View
if (cursor.getCount() > 0)
{
blackNums = new ArrayList<BlackNumInfo>();
}

while (cursor.moveToNext())
{
String num = cursor.getString(0);
int mode = cursor.getInt(1);
int id = cursor.getInt(2);
BlackNumInfo b = new BlackNumInfo(num, mode, id);
blackNums.add(b);
}

db.close();
dbHelper.close();
return blackNums;
}

2. AsyncTask类的内部优化

  • 思路:每次开启一个异步任务查询数据。

  • 重置适配器中,数据容器的值,刷想适配器
  • ListView重绘
/**
* 异步加载
*/
public void fillData()
{
new AsyncTask<String, Integer, String>()
{

// 程序运行前
@Override
protected void onPreExecute()
{
mLoadProgressBar.setVisibility(View.VISIBLE);
// 显示加载进度
super.onPreExecute();
}

// 程序运行时
@Override
protected String doInBackground(String... params)
{
// 查询 前30项的数据: 用子线程 ,
if (mBlackNumList == null)
{
mBlackNumList = dao.getPartBlackNum(limit, offset);
} else
{
// 如果不是第一次查询
if (dao.getPartBlackNum(limit, offset) != null)
{
//把查询到的数据添加到容器中去
mBlackNumList.addAll(dao.getPartBlackNum(limit, offset));
}
}

return null;
}

// 运行后
@Override
protected void onPostExecute(String result)
{
//ProgressBar设置为不可见。
mLoadProgressBar.setVisibility(View.INVISIBLE);

if (adapter == null)
{
//第一次加载,创建适配器适配器
adapter = new BlackNumAdapter(context, mBlackNumList);
mBlackNumListView.setAdapter(adapter);
} else
{
//第二次加载,适配器中容器数量
adapter.setBlackNumList(mBlackNumList);
//发送通知,适配器内容改变,从新加载
adapter.notifyDataSetChanged();
}
}

}.execute();
}

3.Adapter优化分页数据

  • 思路

  • convertView可回收视图的再次利用。
  • holderView类保存每次inflate出来View的子控件并保存在convertViewTag中。
  • 添加一个setDate()方法,重置容器数据

public class BlackNumAdapter extends BaseAdapter
{
List<BlackNumInfo> mBlackNumList;
Context context;
BlackNumDao dao;

public BlackNumAdapter(Context context, List<BlackNumInfo> mBlackNumList)
{
this.context = context;
dao = new BlackNumDao(context);
this.mBlackNumList = mBlackNumList;
}

public void setBlackNumList(List<BlackNumInfo> mBlackNumInfos)
{
this.mBlackNumList = mBlackNumInfos;
}

@Override
public int getCount()
{
return mBlackNumList.size();
}

@Override
public Object getItem(int position)
{
return mBlackNumList.get(position);
}

@Override
public long getItemId(int position)
{
return position;
}

@Override
public View getView(final int position, View convertView, ViewGroup parent)
{
// 适配器的优化
final BlackNumInfo numInfo = mBlackNumList.get(position);
holderView hV = null;
if (convertView == null)
{
hV = new holderView();
convertView = LayoutInflater.from(context).inflate(R.layout.item_blacknum, parent,
false);
hV.modeTextView = (TextView) convertView.findViewById(R.id.tv_mode);
hV.numberTextView = (TextView) convertView.findViewById(R.id.tv_number);
hV.deleteImageView = (ImageView) convertView.findViewById(R.id.iv_deleteNumber);
convertView.setTag(hV);
} else
{
hV = (holderView) convertView.getTag();
}

hV.numberTextView.setText(mBlackNumList.get(position).getNum());

String mode = "全部拦截";
switch (numInfo.getMode())
{
case BlackNumDao.ALL:
mode = "全部链接";
break;
case BlackNumDao.CLL:
mode = "电话拦截";
case BlackNumDao.SMS:
mode = "短信拦截";
default:
break;
}

hV.modeTextView.setText(mode);
hV.deleteImageView.setOnClickListener(new OnClickListener()
{
@Override
public void onClick(View v)
{
// 删除功能
dao.delete(numInfo.getNum() + "");
mBlackNumList.remove(position);
notifyDataSetChanged();
}
});
return convertView;
}

}

/**
* 包装类
* @author H_lang
*
*/
class holderView
{
public TextView modeTextView;
public TextView numberTextView;
public ImageView deleteImageView;
}

4.ListView滚动监听。

  • liseView设置滚动监听。

  • 重置分页查询的开始标度

/**
* 滚动状态改变的时候。
*/
@Override
public void onScrollStateChanged(AbsListView view, int scrollState)
{
// 根据滚动的状态来加载数据。
if (scrollState == OnScrollListener.SCROLL_STATE_TOUCH_SCROLL)
{
// 获得最后一次显示的数据
int lastVisiblePosition = mBlackNumListView.getLastVisiblePosition();
int size = mBlackNumList.size();

if (lastVisiblePosition == size - 1)
{
//设置下次查询的开始位置。
offset = limit + offset;
fillData();
}

}

}

总结

其中还是有许多的缺点。还有许多需要优化的功能。但是对于我现在学习的技术,没有想到更好的方法,希望以后我能继续优化,或者重写listView控件。

ListView优化分页优化的更多相关文章

  1. 转-ListView的性能优化之convertView和viewHolder

    ListView的性能优化之convertView和viewHolder 2014-05-14 参考:http://www.cnblogs.com/xiaowenji/archive/2010/12/ ...

  2. mysql百万级分页优化

    普通分页 数据分页在网页中十分多见,分页一般都是limit start,offset,然后根据页码page计算start , 这种分页在几十万的时候分页效率就会比较低了,MySQL需要从头开始一直往后 ...

  3. 【MySQL】分页优化

    前段时间由于项目的原因,对一个由于分页而造成性能较差的SQL进行优化,现在将优化过程中学习到关于分页优化的知识跟大家简单分享下. 分页不外乎limit,offset,在这两个关键字中,limit其实不 ...

  4. MySQL 百万级分页优化

    MySQL 百万级分页优化 http://www.jb51.net/article/31868.htm 一般刚开始学SQL的时候,会这样写 : , ; 但在数据达到百万级的时候,这样写会慢死 : , ...

  5. 【Android】以BaseAdapter做适配器的ListView及其性能优化

    适配器的Java类 package com.app.adapter; import org.json.JSONArray; import org.json.JSONObject; import and ...

  6. ListView复用和优化详解

    我们每一个Android开发人员对ListView的使用肯定是很熟悉的,然而多少人能真正的懂ListView的缓存机制呢,说白了就是ListView为了提高效率,而内部实现的一种优化,牺牲一点内存.而 ...

  7. 使用Row_Number()分页优化

    记一次SQLServer的分页优化兼谈谈使用Row_Number()分页存在的问题   最近有项目反应,在服务器CPU使用较高的时候,我们的事件查询页面非常的慢,查询几条记录竟然要4分钟甚至更长,而且 ...

  8. MySQL分页优化中的“INNER JOIN方式优化分页算法”到底在什么情况下会生效?

    本文出处:http://www.cnblogs.com/wy123/p/7003157.html 最近无意间看到一个MySQL分页优化的测试案例,并没有非常具体地说明测试场景的情况下,给出了一种经典的 ...

  9. python全栈开发day58-mysql存储过程,权限,索引,慢日志,执行计划,分页优化处理

    1.存储过程 delimiter // create procedure insert_data(in rows int) begin DECLARE n INT DEFAULT 1; drop ta ...

随机推荐

  1. WordPress Cart66 Lite插件HTML注入漏洞

    漏洞名称: WordPress Cart66 Lite插件HTML注入漏洞 CNNVD编号: CNNVD-201310-525 发布时间: 2013-10-23 更新时间: 2013-10-23 危害 ...

  2. Linux kernel 内存损坏漏洞

    漏洞名称: Linux kernel 内存损坏漏洞 CNNVD编号: CNNVD-201310-143 发布时间: 2013-10-11 更新时间: 2013-10-11 危害等级: 中危   漏洞类 ...

  3. android基本的数据库创建和使用

    android的四大组件中就有Content Provider,对其他应用,提供自己的数据,所以,一般情况下,android应用不需要提供content provider. 1.  简单的数据库表单字 ...

  4. 比较详细的利用虚拟机对SD卡FAT32+EXT4+Ext4分区图解教程

    如果大家嫌虚拟机复杂,我这里提供一个我没用虚拟机之前的分区方法:这个方法实际是可行的 我在没有用虚拟机之前,我是这样操作的1.首先在分区软件分好fat32+ext2+ext22.然后用recovery ...

  5. PowerDesigner 的常用小技巧 转

    中小 订阅 修改外键命名规则 选择Database—>Edit Current DBMS选择Scripts->Objects->Reference->ConstName可以发现 ...

  6. TOAD Menu Shortcuts 快捷键

    TOAD Menu Shortcuts Category Command Shortcut Conflict File AWR Browser File Compare Files File DBMS ...

  7. JavaScript---网络编程(12)--DHTML技术演示(5)-form表单验证技术(正则)

    这里不进行很复杂的后台验证以及JavaScript的正则表达式,只是简单的介绍下这个技术,简单的后台接收与跳转,大概了解怎么验证的就可以.具体的技术,我后面还会继续写博客的.本人也还在学习中. 表单验 ...

  8. ASP.NET 路由

    URL 模式可以包含文本值和变量占位符(也称为“URL 参数”).  文本和占位符位于由斜杠 (/) 字符分隔的 URL 段中. 当生成请求时,URL 分析为段和占位符,变量值提供给请求处理程序.  ...

  9. UVA 11624 Fire! (bfs)

    算法指南白书 分别求一次人和火到达各个点的最短时间 #include<cstdio> #include<cstring> #include<queue> #incl ...

  10. 【转】谁说Vim不是IDE?(一)

    谁说Vim不是IDE?(一)   话说操作系统.编程语言和编辑器似乎是程序员永恒的吐槽话题,技术发展了几十年,大家讨论起这几个“之争”还是充满愤怒.津津乐道.有人描述过Emacs和VI程序员,大家沿着 ...