android ListView结合xutils3仿微信实现下拉加载更多
前言:最近涉及到和QQ打交道,定义所有的好友一共只能有300条消息,如果一次性从数据库读取300条或者更多,界面会有细微的卡顿.所以考虑了下分页,第一次进来只显示20条(仿微信),当用户滑到第一条后,如果数据库有消息,则再加载20条.
步骤-问把大象关冰箱,总共分几步?
1.自定义absListview.scrollListerner
核心的东西是监听ListView的scrollListerner,这里采取拿来主义原文连接找不到了,如果原作者看到,请联系青楼爱小生,将您的文章链接更新出来,网上扒了一个挺不错的,大家用的时候实现这个scrollListerner,完善自己的逻辑即可
publicclassMyOnScrollListenerimplementsOnScrollListener{
privateinttotalItemCount;
//ListView最后的item项
privateintlastItem;
//listview第一项
privateintfirstItem;
//用于判断当前是否在加载
privatebooleanisLoading;
//底部加载更多布局
privateViewfooter;
//接口回调的实例
privateOnloadDataListenerlistener;
//数据
privateListdata;
Handlerhandler=newHandler();
publicMyOnScrollListener(Viewfooter,Listdata){
this.footer=footer;
this.data=data;
}
//设置接口回调的实例
publicvoidsetOnLoadDataListener(OnloadDataListenerlistener){
this.listener=listener;
}
/**
*滑动状态变化
*
*@paramview
*@paramscrollState1SCROLL_STATE_TOUCH_SCROLL是拖动2SCROLL_STATE_FLING是惯性滑动0SCROLL_STATE_IDLE是停止,只有当在不同状态间切换的时候才会执行
*/
@Override
publicvoidonScrollStateChanged(AbsListViewview,intscrollState){
//如果数据没有加载,并且滑动状态是停止的,并且滚到了第一个item,可在此做下拉更新或者上拉更新的判断
if(!isLoading&&firstItem==0&&scrollState==SCROLL_STATE_IDLE){
//显示加载更多
footer.setVisibility(View.VISIBLE);
//模拟一个延迟两秒的刷新功能
handler.postDelayed(newRunnable(){
@Override
publicvoidrun(){
if(listener!=null){
//开始加载更多数据
loadMoreData();
//回调设置ListView的数据
listener.onLoadData(data);
//加载完成后操作什么
loadComplete();
}
}
},2000);
}
}
/**
*当加载数据完成后,设置加载标志为false表示没有加载数据了
*并且设置底部加载更多为隐藏
*/
privatevoidloadComplete(){
isLoading=false;
footer.setVisibility(View.GONE);
}
/**
*开始加载更多新数据,这里每次只更新三条数据
*/
privatevoidloadMoreData(){
isLoading=true;
MsgBeanmsg=null;
for(inti=0;i<3;i++){
msg=newMsgBean();
msg.setRemark("Liming"+i);
msg.setMsgID(i);
data.add(stu);
}
}
/**
*监听可见界面的情况
*
*@paramviewListView
*@paramfirstVisibleItem第一个可见的item的索引
*@paramvisibleItemCount可以显示的item的条数
*@paramtotalItemCount总共有多少个item
*/
@Override
publicvoidonScroll(AbsListViewview,intfirstVisibleItem,intvisibleItemCount,inttotalItemCount){
//实现下拉加载
lastItem=firstVisibleItem+visibleItemCount;
//实现上拉加载
firstItem=firstVisibleItem;
//总listView的item个数
this.totalItemCount=totalItemCount;
}
//回调接口
publicinterfaceOnloadDataListener{
voidonLoadData(Listdata);
}
}
2.实现此接口
publicclassListPageActivityextendsActivityimplementsMyOnScrollListener.OnloadDataListener{
@Override
protectedvoidonCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_list_page);
//显示到ListView上
showListView(data);
//自定义的滚动监听事件
MyOnScrollListeneronScrollListener=newMyOnScrollListener(header,data);
//设置接口回调
onScrollListener.setOnLoadDataListener(this);
//设置ListView的滚动监听事件
mListView.setOnScrollListener(onScrollListener);
@Override
publicvoidonLoadData(Listdata){
//加载数据完成后,展示数据到ListView
showListView(data);
}
}
showListView里面无疑是普通的更新adapter的工作
那么我们如何借助xutils的数据库进行分类呢?
3.利用xutils数据库操作进行分页处理
首先,我们理一下思路,上面我们已经实现了上拉的回调,在此回调中把新来的数据加载到adapter即可.
下文db是Dbmanager的实例
/** *当前屏幕显示的消息数量 */ privateintMAX_MSG_NUMBER=20;
privateListgetDataFromDb(){ List>dbSize=db.selector(MsgBean.class).where(WhereBuilder.b("id","=",400)).findAll();//记得捕获null指针和DbException异常 //如果数据库比我们显示的页数小,则不偏移,否则,偏移到我们需要显示的位置 if(dbSize.size() datas=db.selector(MsgBean.class).where(WhereBuilder.b("id","=",400)).limit(MAX_MSG_NUMBER) .offset(indexOffset).findAll(); returndatas; }
这里解释一下
db.selector(MsgBean.class).where(WhereBuilder.b("id","=",400)).limit(MAX_MSG_NUMBER).offset(indexOffset).findAll();是我们实现分页的关键
.limit是我们定义的分页大小
.offset偏移量,我们数据库的大小是不变的,如果不定义偏移量,那么我们定义的分页大小每次只从0取到19.假设数据库中有21条数据,那么我们需要从1取到20,而不是0到19,所以偏移1.
然后我们在loadMoreData中
MAX_MSG_NUMBER+=MAX_MSG_NUMBER; getDataFromDb();
将大小自加,即完成加载更多的功能,在onLoadData(List
后面贴上我对xutils数据库操作的封装,还有很多不完善之处
/**
*数据库xutils用法
*@author青楼爱小生
*/
publicclassDbUtil{
privatestaticfinalStringTAG=DbUtil.class.getName();
privatestaticDbUtildbUtil;
privateDbManagerdb;
privateDbUtil(){
db=x.getDb(MyApplication.getInstance().daoConfig);
}
publicstaticDbUtilgetInstance(){
if(dbUtil==null){
synchronized(DbUtil.class){
if(dbUtil==null){
dbUtil=newDbUtil();
}
}
}
returndbUtil;
}
/**
*增加数据
*@paramlist
*@throwsDbException
*/
publicvoidaddMsgList(Listlist){
try{
db.saveOrUpdate(list);
}catch(DbExceptione){
e.printStackTrace();
LogHelper.e(TAG,e.getMessage());
}
}
/**
*增加一条数据
*@paramnode
*@throwsDbException
*/
publicvoidaddMsgToDb(MsgBeannode){
try{
db.saveOrUpdate(node);
}catch(DbExceptione){
e.printStackTrace();
LogHelper.e(TAG,e.getMessage());
}
}
/**
*删除表中所有数据
*@paramcls创建的表的映射
*@throwsDbException
*/
publicvoiddeleteAll(Classcls){
try{
db.delete(cls);
}catch(DbExceptione){
LogHelper.e(TAG,e.getMessage());
e.printStackTrace();
}
}
/**
*删除第一条数据
*@paramcls
*/
@SuppressWarnings("unchecked")
publicvoiddeleteFirst(Classcls){
try{
db.delete(db.findFirst(cls));
}catch(DbExceptione){
//TODOAuto-generatedcatchblock
e.printStackTrace();
}
}
/**
*查询表中所有数据
*@throwsDbException
*/
@SuppressWarnings("unchecked")
publicList>findAll(Classcls){
try{
returndb.findAll(cls)==null?Collections.emptyList():db.findAll(cls);
}catch(DbExceptione){
e.printStackTrace();
LogHelper.e(TAG,e.getMessage());
returnCollections.emptyList();
}
}
/**
*//添加查询条件进行查询
Listall=db.selector(ChildInfo.class).where("id",">",2).and("id","<",4).findAll();
*@return搜索指定条件的数据
*/
@SuppressWarnings("unchecked")
publicList>findDataByWhere(Classcls,WhereBuilderformat){
try{
returndb.selector(cls).where(format).findAll()==null?
Collections.emptyList():db.selector(cls).where(format).findAll();
}catch(DbExceptione){
LogHelper.e(TAG,e.getMessage());
e.printStackTrace();
returnCollections.emptyList();
}
}
/**
*添加查询条件进行查询
*@paramcls表映射
*@paramstrselect语句
*@paramformatwhere语句
*@returnListDbModelkey为数据库列名value为值
*eg:(Selector.from(Parent.class)
.where("id","<",54)
.and(WhereBuilder.b("age",">",20).or("age","<",30))
.orderBy("id")
.limit(pageSize).offset(pageSize*pageIndex));
*
*
*
*/
@SuppressWarnings("unchecked")
publicSelector>findDataBySelector(Classcls,WhereBuilderformat){
try{
returndb.selector(cls).where(format);
}catch(DbExceptione){
//TODOAuto-generatedcatchblock
e.printStackTrace();
}
returnnull;
}
}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。