国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統(tǒng) > Android > 正文

Android ListView分頁功能實(shí)現(xiàn)方法

2019-12-12 06:26:56
字體:
供稿:網(wǎng)友

通過本次小Demo我學(xué)到了:

1、ListView的小小的一個(gè)分頁功能
2、加深了對自定義控件的理解
3、對ListView的優(yōu)化
4、對BaseAdapter的使用
5、自定義Adapter
6、接口的回調(diào)

要實(shí)現(xiàn)下面的效果--當(dāng)拖動(dòng)ListView到底部的時(shí)候,顯示一個(gè)ProgressBar和一個(gè)"正在加載..."的TextView。并且過兩秒鐘后,在下面加載出新的數(shù)據(jù)。項(xiàng)目的目錄結(jié)構(gòu)和程序要實(shí)現(xiàn)的效果如下:

                 

首先是布局部分:

我為了實(shí)現(xiàn)此效果,首先在布局文件中新建了一個(gè)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>

然后新建了一個(gè)item.xml用于作為ListView的子項(xiàng):

<?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>

最后在主布局文件中添加了一個(gè)自定義的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>

然后為了實(shí)現(xiàn)ListView的這種效果,我們需要一個(gè)自定義的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;// 最后一個(gè)可見的Item int totalItemCount;// Item的總數(shù)量 boolean isLoading; // 正在加載 ILoadListener iLoadListener; public LoadListView(Context context, AttributeSet attrs, int defStyle) {  super(context, attrs, defStyle);  // TODO 自動(dòng)生成的構(gòu)造函數(shù)存根  initView(context); } public LoadListView(Context context, AttributeSet attrs) {  super(context, attrs);  // TODO 自動(dòng)生成的構(gòu)造函數(shù)存根  initView(context); } public LoadListView(Context context) {  super(context);  // TODO 自動(dòng)生成的構(gòu)造函數(shù)存根  initView(context); } /***  * 添加底部提示加載布局到listView  *   * @param context  */ public void initView(Context context) {  LayoutInflater inflater = LayoutInflater.from(context);  footer = inflater.inflate(R.layout.footer_layout, null);  // 初始時(shí)候讓底部布局不可見  footer.findViewById(R.id.load_layout).setVisibility(View.GONE);  this.addFooterView(footer);  this.setOnScrollListener(this); } @Override public void onScrollStateChanged(AbsListView view, int scrollState) {  // TODO 自動(dòng)生成的方法存根  // 當(dāng)總共的Item數(shù)量等于最后一個(gè)Item的位置,并且滾動(dòng)停止時(shí)  if (totalItemCount == lastVisiableItem    && scrollState == SCROLL_STATE_IDLE) {   if (!isLoading) {    isLoading = true;    footer.findViewById(R.id.load_layout).setVisibility(      View.VISIBLE);    //加載更多    iLoadListener.onLoad();   }  } }  /** *firstVisibleItem 第一個(gè)可見Item的位置 *visibleItemCount 可見的Item的數(shù)量 *totalItemCount  Item的總數(shù)量 **/ @Override public void onScroll(AbsListView view, int firstVisibleItem,   int visibleItemCount, int totalItemCount) {  // TODO 自動(dòng)生成的方法存根  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; } //加載更多數(shù)據(jù)回調(diào)接口 public interface ILoadListener {  public void onLoad(); }}

我們自定義的ListView繼承自ListView,并實(shí)現(xiàn)其中父類的三個(gè)構(gòu)造方法,為了將底部我們想要的布局加載到ListView中來,我們自定義了一個(gè)initView方法,用于找到并實(shí)例化footer_layout.xml使其添加到ListView底部。在父類的三個(gè)構(gòu)造方法中添加初始化方法initView(),在initView方法的最后還要調(diào)用ListView的addFooterView(View)方法,將底部布局add進(jìn)來。由于在ListView剛加載進(jìn)來的時(shí)候我們不想顯示這個(gè)footer,所以要設(shè)置它的Visible為GONE。想要實(shí)現(xiàn)ListView拉到底部的時(shí)候才顯示footer,要實(shí)現(xiàn)ListView的OnScrollListener接口,并實(shí)現(xiàn)其父類中的兩個(gè)方法。在OnScrollStateChanged()方法中判斷是否滾動(dòng)到底部(我們定義了一個(gè)全局變量lastVisibleItem=firstVisibleItem+VisibleItemCount,若此值和totalItemCount相等,則證明滾動(dòng)到ListView的底端了)和此時(shí)ListView是否停止?jié)L動(dòng)(scrollState=SCROLL_STATE_IDLE)。

為了向ListView中添加數(shù)據(jù)我們定義了一個(gè)實(shí)體類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; } }

之后我們?yōu)長istView定義了一個(gè)數(shù)據(jù)適配器MyAdapter,繼承自BaseAdapter,并實(shí)現(xiàn)其中的四個(gè)方法,在其中我們主要實(shí)現(xiàn)數(shù)據(jù)的填充:

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;   //構(gòu)造函數(shù)中傳入了list列表項(xiàng)和初始化LayoutInflater public MyAdapter(Context context,ArrayList<ApkEntity> list) {  this.list=list;  this.inflater=LayoutInflater.from(context); } //得到list的長度(是程序在加載顯示到UI上是就要先讀取的,這里獲得的值決定了ListView顯示多少行) @Override public int getCount() {  // TODO 自動(dòng)生成的方法存根  return list.size(); } //得到list中指定位置的data(根據(jù)ListView所在的位置返回View) @Override public Object getItem(int position) {  // TODO 自動(dòng)生成的方法存根  return list.get(position); }  //根據(jù)ListView位置得到數(shù)據(jù)源集合中的ID @Override public long getItemId(int position) {  // TODO 自動(dòng)生成的方法存根  return position; } //***最主要,決定ListView的界面樣式 @Override public View getView(int position, View convertView, ViewGroup parent) {  // TODO 自動(dòng)生成的方法存根  //從list中獲取實(shí)體  ApkEntity entity=list.get(position);  //使用ViewHolder的目的是為了使每次在getView的時(shí)候不是每次都findViewById()來獲取控件實(shí)例  ViewHolder holder;  /**   * convertView:The old View to reuses   * 用于將之前加載好的布局緩存,以便之后可以重用   */  //為了避免重復(fù)加載布局,僅僅在convertView為空的時(shí)候才使用LayoutInflate加載布局  if(convertView==null){   holder=new ViewHolder();   //找到并將layout轉(zhuǎn)換為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; }  //布局改變時(shí)用來刷新ListView public void onDateChanged(ArrayList<ApkEntity> list){  this.list=list;  this.notifyDataSetChanged(); }}

在這個(gè)自定義Adapter中最主要的就是getView()方法,它決定了ListView的每項(xiàng)的布局(長什么樣),在getView()方法中,為了優(yōu)化ListView的運(yùn)行效率,使得不是每次Item創(chuàng)建的時(shí)候都要findViewById()來實(shí)例化控件,我們定義了一個(gè)ViewHolder的內(nèi)部類,用來對控件的實(shí)例進(jìn)行緩存,在類中聲明了Item布局中的布局控件。因?yàn)間etView()方法每次都將布局重新加載了一遍,所以在ListView快速滾動(dòng)的時(shí)候就會(huì)成為性能的瓶頸。所以用到了getView()方法中的convertView參數(shù),這個(gè)參數(shù)用于將之前加載好的布局進(jìn)行緩存,以便之后可以重新使用。通過上面的代碼可以看到,如果convertView 為空的時(shí)候,我們就使用LayoutInflate加載布局并實(shí)例化Item中的控件,還要調(diào)用View的setTag()方法,將ViewHolder對象存儲(chǔ)在convertViewu 中。這樣,當(dāng)convertView不為空的時(shí)候,則直接調(diào)用View的getTag()方法,把ViewHolder直接取出,這樣所有的控件的實(shí)例都緩存在了ViewHolder里,就沒有必要每次都對控件進(jìn)行findViewById()了。

1.使用convertView參數(shù):避免重復(fù)加載布局,用他來對之前加載過的布局進(jìn)行緩存。

2.使用ViewHolder:避免每次getView()的時(shí)候都對控件進(jìn)行實(shí)例化,用這個(gè)類完成對控件實(shí)例化的緩存。

然后我們需要完成在MainActivity中對LoadListView的實(shí)例化,和Mydapter的實(shí)例化,向?qū)嶓w類中添加數(shù)據(jù)并使adapter和ListView適配完成填充數(shù)據(jù):

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 自動(dòng)生成的方法存根  for (int i = 0; i < 10; i++) {   ApkEntity entity=new ApkEntity();   entity.setName("大毛");   entity.setInfo("我是一只pig");   list.add(entity);  }  }  private void getOnLoadDate() {  // TODO 自動(dòng)生成的方法存根  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 自動(dòng)生成的方法存根  //用現(xiàn)線程來控制隔多少秒之后獲取數(shù)據(jù),然后設(shè)置到ListView上(正常情況下不需要加,只是為了看出來這個(gè)延時(shí)的效果)  Handler handler=new Handler();  handler.postDelayed(new Runnable() {    @Override   public void run() {    // TODO 自動(dòng)生成的方法存根    getOnLoadDate();    showListView(list);    //通知ListView加載完畢    lv.loadComplete();   }  }, 2000);  }}

MainActivity中主要需要注意的就是showListView()方法,在該方法中我們判斷了一下adapter是否為空,若adapter為空,則實(shí)例化listview,實(shí)例化adapter等等一系列操作,否則調(diào)用MyAdapter的onDateChanged()方法(此方法中調(diào)用了Adapter的notifyDataSetChanged()方法此方法用于ListView發(fā)生變化時(shí)更新UI)。由于要在監(jiān)聽到滑動(dòng)到ListView底部的時(shí)候加載新的數(shù)據(jù),所以在LoadListView類中實(shí)現(xiàn)一個(gè)隊(duì)MainActivoity的回調(diào),在LoadListView中寫一個(gè)回調(diào)接口ILoadListener(),在其中實(shí)現(xiàn)一個(gè)onLoad()方法,在MainActivity中實(shí)現(xiàn)這個(gè)接口,重寫onLoad()方法,在其中 實(shí)現(xiàn)想要實(shí)現(xiàn)的其他方法,比如新數(shù)據(jù)的加載和UI的刷新展示,最后,刷新加載完新的數(shù)據(jù)后,要將footer隱藏,所以執(zhí)行LoadListView中的loadComplete()方法。

至此,整個(gè)小Demo的學(xué)習(xí)基本完成,其中還有些知識(shí)不太懂,比如說接口的回調(diào),自定義控件部分等等,還需要加深練習(xí)。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 吉木萨尔县| 仁寿县| 临漳县| 香格里拉县| 高邑县| 郴州市| 库伦旗| 沈阳市| 通州区| 仲巴县| 莱州市| 长治市| 高州市| 普兰店市| 兴隆县| 古丈县| 镇雄县| 保亭| 河西区| 西盟| 禄丰县| 同仁县| 赤城县| 建昌县| 通道| 南昌市| 天祝| 耒阳市| 文化| 汨罗市| 安多县| 迁西县| 云阳县| 新和县| 陆川县| 吉安市| 宁安市| 新田县| 肃北| 长垣县| 蕲春县|