Android ListView分页功能实现方法_Android

通过本次小Demo我学到了:

1、ListView的小小的一个分页功能
2、加深了对自定义控件的理解
3、对ListView的优化
4、对BaseAdapter的使用
5、自定义Adapter
6、接口的回调

要实现下面的效果--当拖动ListView到底部的时候,显示一个ProgressBar和一个"正在加载..."的TextView。并且过两秒钟后,在下面加载出新的数据。项目的目录结构和程序要实现的效果如下:

                 

首先是布局部分:

我为了实现此效果,首先在布局文件中新建了一个footer_layout.xml的布局文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:orientation="vertical" >
 <LinearLayout
  android:id="@+id/load_layout"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:orientation="horizontal"
  android:paddingTop="10dip"
  android:paddingBottom="10dip"
  android:gravity="center"
  >
  <ProgressBar
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   style="?android:attr/progressBarStyleSmall"
   android:background="#ff0000"
   />
  <TextView
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:text="正在加载..."
   />

 </LinearLayout>

</LinearLayout>

然后新建了一个item.xml用于作为ListView的子项:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:orientation="vertical" >

 <TextView
  android:id="@+id/tv1"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:text="哈哈哈" />
 <TextView
  android:id="@+id/tv2"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:text="嘎嘎嘎嘎嘎"
 />
</LinearLayout>

最后在主布局文件中添加了一个自定义的ListView控件:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 >

 <com.lx.loadListView.LoadListView
  android:id="@+id/list"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:layout_alignParentTop="true"
  android:layout_centerHorizontal="true"
  android:cacheColorHint="#00000000" >
 </com.lx.loadListView.LoadListView>

</RelativeLayout>

然后为了实现ListView的这种效果,我们需要一个自定义的ListView,并在上面的布局文件中引用我们自定义的ListView,代码如下:

package com.lx.loadListView;

import com.example.listviewloaddemo.R;

import android.content.Context;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.AbsListView;
import android.widget.ListView;
import android.widget.AbsListView.OnScrollListener;

public class LoadListView extends ListView implements OnScrollListener {

 View footer;
 int lastVisiableItem;// 最后一个可见的Item
 int totalItemCount;// Item的总数量
 boolean isLoading; // 正在加载
 ILoadListener iLoadListener;

 public LoadListView(Context context, AttributeSet attrs, int defStyle) {
  super(context, attrs, defStyle);
  // TODO 自动生成的构造函数存根
  initView(context);
 }

 public LoadListView(Context context, AttributeSet attrs) {
  super(context, attrs);
  // TODO 自动生成的构造函数存根
  initView(context);
 }

 public LoadListView(Context context) {
  super(context);
  // TODO 自动生成的构造函数存根
  initView(context);
 }

 /***
  * 添加底部提示加载布局到listView
  *
  * @param context
  */
 public void initView(Context context) {
  LayoutInflater inflater = LayoutInflater.from(context);
  footer = inflater.inflate(R.layout.footer_layout, null);
  // 初始时候让底部布局不可见
  footer.findViewById(R.id.load_layout).setVisibility(View.GONE);
  this.addFooterView(footer);
  this.setOnScrollListener(this);
 }

 @Override
 public void onScrollStateChanged(AbsListView view, int scrollState) {
  // TODO 自动生成的方法存根
  // 当总共的Item数量等于最后一个Item的位置,并且滚动停止时
  if (totalItemCount == lastVisiableItem
    && scrollState == SCROLL_STATE_IDLE) {
   if (!isLoading) {
    isLoading = true;
    footer.findViewById(R.id.load_layout).setVisibility(
      View.VISIBLE);
    //加载更多
    iLoadListener.onLoad();
   }
  }
 }

 /**
 *firstVisibleItem 第一个可见Item的位置
 *visibleItemCount 可见的Item的数量
 *totalItemCount  Item的总数量
 **/
 @Override
 public void onScroll(AbsListView view, int firstVisibleItem,
   int visibleItemCount, int totalItemCount) {
  // TODO 自动生成的方法存根
  this.lastVisiableItem = firstVisibleItem + visibleItemCount;
  this.totalItemCount = totalItemCount;
 }

 //加载完毕将footer隐藏
 public void loadComplete(){
  isLoading=false;
  footer.findViewById(R.id.load_layout).setVisibility(View.GONE);
 }

 public void setInterface(ILoadListener iLoadListener) {
  this.iLoadListener = iLoadListener;
 }

 //加载更多数据回调接口
 public interface ILoadListener {
  public void onLoad();
 }

}

我们自定义的ListView继承自ListView,并实现其中父类的三个构造方法,为了将底部我们想要的布局加载到ListView中来,我们自定义了一个initView方法,用于找到并实例化footer_layout.xml使其添加到ListView底部。在父类的三个构造方法中添加初始化方法initView(),在initView方法的最后还要调用ListView的addFooterView(View)方法,将底部布局add进来。由于在ListView刚加载进来的时候我们不想显示这个footer,所以要设置它的Visible为GONE。想要实现ListView拉到底部的时候才显示footer,要实现ListView的OnScrollListener接口,并实现其父类中的两个方法。在OnScrollStateChanged()方法中判断是否滚动到底部(我们定义了一个全局变量lastVisibleItem=firstVisibleItem+VisibleItemCount,若此值和totalItemCount相等,则证明滚动到ListView的底端了)和此时ListView是否停止滚动(scrollState=SCROLL_STATE_IDLE)。

为了向ListView中添加数据我们定义了一个实体类Apk_Entity:

package com.lx.entity;

public class ApkEntity {

 private String name;
 private String info;
 public String getName() {
  return name;
 }
 public void setName(String name) {
  this.name = name;
 }
 public String getInfo() {
  return info;
 }
 public void setInfo(String info) {
  this.info = info;
 }

}

之后我们为ListView定义了一个数据适配器MyAdapter,继承自BaseAdapter,并实现其中的四个方法,在其中我们主要实现数据的填充:

package com.lx.adapter;

import java.util.ArrayList;

import com.example.listviewloaddemo.R;
import com.lx.entity.ApkEntity;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;

public class MyAdapter extends BaseAdapter {

 ArrayList<ApkEntity> list;
 LayoutInflater inflater;

 //构造函数中传入了list列表项和初始化LayoutInflater
 public MyAdapter(Context context,ArrayList<ApkEntity> list) {
  this.list=list;
  this.inflater=LayoutInflater.from(context);
 }

 //得到list的长度(是程序在加载显示到UI上是就要先读取的,这里获得的值决定了ListView显示多少行)
 @Override
 public int getCount() {
  // TODO 自动生成的方法存根
  return list.size();
 }

 //得到list中指定位置的data(根据ListView所在的位置返回View)
 @Override
 public Object getItem(int position) {
  // TODO 自动生成的方法存根
  return list.get(position);
 }

 //根据ListView位置得到数据源集合中的ID
 @Override
 public long getItemId(int position) {
  // TODO 自动生成的方法存根
  return position;
 }

 //***最主要,决定ListView的界面样式
 @Override
 public View getView(int position, View convertView, ViewGroup parent) {
  // TODO 自动生成的方法存根
  //从list中获取实体
  ApkEntity entity=list.get(position);
  //使用ViewHolder的目的是为了使每次在getView的时候不是每次都findViewById()来获取控件实例
  ViewHolder holder;
  /**
   * convertView:The old View to reuses
   * 用于将之前加载好的布局缓存,以便之后可以重用
   */
  //为了避免重复加载布局,仅仅在convertView为空的时候才使用LayoutInflate加载布局
  if(convertView==null){
   holder=new ViewHolder();
   //找到并将layout转换为View
   convertView=inflater.inflate(R.layout.item, null);
   holder.tv_name=(TextView) convertView.findViewById(R.id.tv1);
   holder.tv_info=(TextView) convertView.findViewById(R.id.tv2);
   convertView.setTag(holder);
  }else{
   holder=(ViewHolder) convertView.getTag();
  }
  holder.tv_name.setText(entity.getName());
  holder.tv_info.setText(entity.getInfo());
  return convertView;
 }

 class ViewHolder{
  TextView tv_name,tv_info;
 }

 //布局改变时用来刷新ListView
 public void onDateChanged(ArrayList<ApkEntity> list){
  this.list=list;
  this.notifyDataSetChanged();
 }

}

在这个自定义Adapter中最主要的就是getView()方法,它决定了ListView的每项的布局(长什么样),在getView()方法中,为了优化ListView的运行效率,使得不是每次Item创建的时候都要findViewById()来实例化控件,我们定义了一个ViewHolder的内部类,用来对控件的实例进行缓存,在类中声明了Item布局中的布局控件。因为getView()方法每次都将布局重新加载了一遍,所以在ListView快速滚动的时候就会成为性能的瓶颈。所以用到了getView()方法中的convertView参数,这个参数用于将之前加载好的布局进行缓存,以便之后可以重新使用。通过上面的代码可以看到,如果convertView 为空的时候,我们就使用LayoutInflate加载布局并实例化Item中的控件,还要调用View的setTag()方法,将ViewHolder对象存储在convertViewu 中。这样,当convertView不为空的时候,则直接调用View的getTag()方法,把ViewHolder直接取出,这样所有的控件的实例都缓存在了ViewHolder里,就没有必要每次都对控件进行findViewById()了。

1.使用convertView参数:避免重复加载布局,用他来对之前加载过的布局进行缓存。

2.使用ViewHolder:避免每次getView()的时候都对控件进行实例化,用这个类完成对控件实例化的缓存。

然后我们需要完成在MainActivity中对LoadListView的实例化,和Mydapter的实例化,向实体类中添加数据并使adapter和ListView适配完成填充数据:

package com.example.listviewloaddemo;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import com.lx.adapter.MyAdapter;
import com.lx.entity.ApkEntity;
import com.lx.loadListView.LoadListView;
import com.lx.loadListView.LoadListView.ILoadListener;

import android.os.Bundle;
import android.os.Handler;
import android.app.Activity;
import android.util.Log;
import android.view.Menu;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.SimpleAdapter;

public class MainActivity extends Activity implements ILoadListener {

 private LoadListView lv;
 private ArrayList<ApkEntity> list=new ArrayList<ApkEntity>();
 private MyAdapter myAdapter;
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  getDate();
  showListView(list);

 }

 private void getDate() {
  // TODO 自动生成的方法存根
  for (int i = 0; i < 10; i++) {
   ApkEntity entity=new ApkEntity();
   entity.setName("大毛");
   entity.setInfo("我是一只pig");
   list.add(entity);
  }
 }

 private void getOnLoadDate() {
  // TODO 自动生成的方法存根
  for (int i = 0; i < 2; i++) {
   ApkEntity entity=new ApkEntity();
   entity.setName("小毛");
   entity.setInfo("我是一只dog");
   list.add(entity);
  }
 }

 private void showListView(ArrayList<ApkEntity> list) {
  if(myAdapter==null){
   lv = (LoadListView) findViewById(R.id.list);
   lv.setInterface(this);
   Log.d("SetInterface--->>", this.toString());
   myAdapter=new MyAdapter(this, list);
   lv.setAdapter(myAdapter);
  }else{
   myAdapter.onDateChanged(list);
  }
 }

 @Override
 public void onLoad() {
  // TODO 自动生成的方法存根
  //用现线程来控制隔多少秒之后获取数据,然后设置到ListView上(正常情况下不需要加,只是为了看出来这个延时的效果)
  Handler handler=new Handler();
  handler.postDelayed(new Runnable() {
   @Override
   public void run() {
    // TODO 自动生成的方法存根
    getOnLoadDate();
    showListView(list);
    //通知ListView加载完毕
    lv.loadComplete();
   }
  }, 2000);
 }

}

MainActivity中主要需要注意的就是showListView()方法,在该方法中我们判断了一下adapter是否为空,若adapter为空,则实例化listview,实例化adapter等等一系列操作,否则调用MyAdapter的onDateChanged()方法(此方法中调用了Adapter的notifyDataSetChanged()方法此方法用于ListView发生变化时更新UI)。由于要在监听到滑动到ListView底部的时候加载新的数据,所以在LoadListView类中实现一个队MainActivoity的回调,在LoadListView中写一个回调接口ILoadListener(),在其中实现一个onLoad()方法,在MainActivity中实现这个接口,重写onLoad()方法,在其中 实现想要实现的其他方法,比如新数据的加载和UI的刷新展示,最后,刷新加载完新的数据后,要将footer隐藏,所以执行LoadListView中的loadComplete()方法。

至此,整个小Demo的学习基本完成,其中还有些知识不太懂,比如说接口的回调,自定义控件部分等等,还需要加深练习。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索android
, listview
分页
listview实现分页加载、android listview分页、android listview实现、listview置顶功能实现、android实现分页,以便于您获取更多的相关知识。

时间: 2024-09-13 07:13:43

Android ListView分页功能实现方法_Android的相关文章

Android ListView分页功能实现方法

通过本次小Demo我学到了: 1.ListView的小小的一个分页功能 2.加深了对自定义控件的理解 3.对ListView的优化 4.对BaseAdapter的使用 5.自定义Adapter 6.接口的回调 要实现下面的效果--当拖动ListView到底部的时候,显示一个ProgressBar和一个"正在加载..."的TextView.并且过两秒钟后,在下面加载出新的数据.项目的目录结构和程序要实现的效果如下: 首先是布局部分: 我为了实现此效果,首先在布局文件中新建了一个foote

android实现倒计时功能的方法_Android

前言   在打开爱奇艺等app的欢迎界面的时候,右上角有一个倒计时的控件.倒计时完了以后进入主界面.现在我们来实现这个功能.  方法一: 利用java的类Timer,TimerTask还有android的Handler 界面welcome_activity.xml  <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.an

Android ListView分页简单实现

Android ListView分页简单实现 分页,开发应用中必不可少.那么,现在就来实现分页功能. 首先来想想实现它要有哪些步骤, 1, 实现的组件, 2.初始化第一页数据, 3,底部布局 , 4,加载数据的条件 5.获取下一页的数据. 有了思路,我们一步步来实现就行了.先来想想我们用什么组件实现,我们知道列表UI我们常用ListView或者RecyclerView,初始化数据,我们就在通过一个for循环来准备数据,底部布局我们直接使ProgressBar控件和一个TextView来显示就可以

getview方法不执行- android listview自定义adapter,getview方法不显示

问题描述 android listview自定义adapter,getview方法不显示 adapter的getCount值不为0,但是不执行getview方法,在ErrorLog中打印出来的log在下面贴出来了,希望路过的大牛们帮忙看看 解决方案 问题找到了,是我的listView的布局控件不正确,已经改好了 解决方案二: android listView 重写Adapter方法getView的性能优化问题 解决方案三: 空指针异常,找找在哪个位置. 解决方案四: log和问的错没什么关系 解

Android实现登录功能demo示例_Android

本文实例讲述了Android实现登录功能的方法.分享给大家供大家参考,具体如下: 安卓,在小编实习之前的那段岁月里面,小编都没有玩儿过,如果说玩儿过,那就是安卓手机了,咳咳,敲登录的时候有种特别久违的熟悉,这种熟悉的感觉就和当时敲机房收费系统一样,那叫一个艰难啊,不过小编相信,在小编的IT成长之路上,正是因为有了这些艰难险阻陪伴着小编一起成长,才让小编更加勇敢坚强,勇敢的面对一个又一个bug,坚强的敲完一行行代码,经过了几天的研究登录一条线的功能已经实现,现在小编就来简单的总结一下,还请小伙伴们

thinkPHP多表查询及分页功能实现方法示例

本文实例讲述了thinkPHP多表查询及分页功能实现方法.分享给大家供大家参考,具体如下: 项目业务逻辑为:教师上传试卷,设置答题卡,发布答题卡给相关的班级或群组,只有试卷关联的答题卡发布后,该试卷才能在系统试卷中搜索到,同时其他的老师也可以收藏.在前端的收藏模块中,有个业务是给个input框以提供搜索功能给用户,但是在事先设计的搜索表中,只有一处试卷ID是和试卷表关联的,如果用户搜索试卷题目那岂不要两表查询了,一开始我想到的方法是在收藏表中多加个字段,也就是把试卷题目的字段添加到收藏表中,业务

thinkPHP5分页功能实现方法分析

本文实例讲述了thinkPHP5分页功能实现方法.分享给大家供大家参考,具体如下: 其实分页自身的内容也不是很多.不过牵扯到样式的问题感觉挺烦.于是找到了分页类看了一下.把大体的结构说一下.如果有需要修改页面样式的可以自行修改样式.最好提前备份,防止意外. 首先是分页的调用,tp5的调用相对非常容易 $mod = new \app\index\model\Blogmsg(); $mo = $mod->paginate(1,14); $this->assign('list', $mo); //

Android ListView填充数据的方法

Android ListView填充数据的方法 因为多人开发,为了是自己开发的模块方便融合到主框架中,同时也为了减小apk的大小,要求尽可能少的使用xml的布局文件,开发中需要在ListView中显示数据,网上查到的几乎所有的示例,都是通过xml文件来为ListView的Item提供布局样式,甚是不方便. 能不能将自己通过代码创建的布局(如View,LinearLayout)等动态的布局到ListView呢?当然可以. 为了给ListView提供数据,我们需要为其设置一个适配,我们可以从Base

Android提高之SQLite分页表格实现方法_Android

继前一篇文章讲到Android上的SQLite分页读取,其功能只是用文本框显示数据而已.本文就讲得更加深入些,实现并封装一个SQL分页表格控件,不仅支持分页还是以表格的形式展示数据. 先来看看本文程序运行的动画如下图所示: 这个SQL分页表格控件主要分为"表格区"和"分页栏"这两部分,这两部分都是基于GridView实现的.网上介绍Android上实现表格的DEMO一般都用ListView.ListView与GridView对比,ListView最大的优势是格单元的