当前位置:首页 > 开发 > 移动开发 > 正文

Android ListView的OnItemClickListener详解--id和position的区别

发表于: 2012-06-12   作者:矮蛋蛋   来源:转载   浏览次数:
摘要: 我们在使用ListView的时候,一般都会为ListView添加一个响应事件android.widget.AdapterView.OnItemClickListener。本文主要在于对OnItemClickListener的position和id参数做详细的解释,我相信有些人在这上面走了些弯路。 先来看一下官方的文档 position The position of the view in
我们在使用ListView的时候,一般都会为ListView添加一个响应事件android.widget.AdapterView.OnItemClickListener。本文主要在于对OnItemClickListener的position和id参数做详细的解释,我相信有些人在这上面走了些弯路。

先来看一下官方的文档
position The position of the view in the adapter.
id The row id of the item that was clicked.
而这两行字并没有解释清楚position和id的区别。另外,我们还有个Adapter的getView方法。
public abstract View getView (int position, View convertView, ViewGroup parent)

这里也有一个position。

初步接触ListView的同学,一般会直接继承ArrayAdapter,然后(比如我),就想当然的认为OnItemClick的position和getView的position是一样的啊。于是我们就getItem(position)来获取相应的数据。

那么这段代码有没有错呢?如果有错的话,在什么情况会出错呢?
第一个问题的答案是,当我们为ListView添加headerView或者footerView之后,这段代码就不一定是我们想要的了。

出现问题的原因在于,当我们为ListView添加headerView或者footerView之后,ListView在setAdapter时,做了一些事情,这导致,Adapter和OnItemClickListener中的position含义发生了变化。

我们可以来看看ListView中setAdapter的实现
437  public void setAdapter(ListAdapter adapter) {
438      if (mAdapter != null && mDataSetObserver != null) {
439          mAdapter.unregisterDataSetObserver(mDataSetObserver);
440      }
442      resetList();
443      mRecycler.clear();
445      if (mHeaderViewInfos.size() > 0|| mFooterViewInfos.size() > 0) {
446          mAdapter = new HeaderViewListAdapter(mHeaderViewInfos, mFooterViewInfos, adapter);
447      } else {
448          mAdapter = adapter;
449      }
可以看出,如果这个ListView存在headerView或者footerView的话,那么会在我们传入的adapter外面在封装一层HeaderViewListAdapter,这是一个专门用来自动处理headerView和footerView的adapter。在ListView中,本身不区分headerView,footerView。ListView可以理解成是只负责管理一组View的数组的UI(ViewGroup),headerView和footerView都委托给HeaderViewListAdapter来处理。(从这里也可以看到为什么API文档中提到,addFooterView和addHeaderView要在setAdapter函数之前调用,如果在之后调用,那么就不会生成HeaderViewListAdapter,从而导致显示不出headerView和footerView)。

回到开头的问题,position和id有啥区别。为此,我们找一下position和id是怎么传进来的。
OnItemClickListener在android.widget.AdapterView的public boolean performItemClick(View view, int position, long id)函数中被调用。
performItemClick在android.widget.AbsListView.PerformClick.run() 中被调用
2497  private class PerformClick extends WindowRunnnable implements Runnable {
2498      int mClickMotionPosition;
2500      public void run() {
2501          // The data has changed since we posted this action in the event queue,
2502          // bail out before bad things happen
2503          if (mDataChanged) return;
2505          final ListAdapter adapter = mAdapter;
2506          final int motionPosition = mClickMotionPosition;
2507          if (adapter != null && mItemCount > 0 &&
2508                  motionPosition != INVALID_POSITION &&
2509                  motionPosition < adapter.getCount() && sameWindow()) {
2510              final View view = getChildAt(motionPosition - mFirstPosition);
2511              // If there is no view, something bad happened (the view scrolled off the
2512              // screen, etc.) and we should cancel the click
2513              if (view != null) {
2514                  performItemClick(view, motionPosition, adapter.getItemId(motionPosition));
2515              }
2516          }
2517      }
2518  }
可以看到,position事实上就是ListView中被点击的view的位置。注意,在ListView中是不负责处理headerView和footViewer的,所以,这个位置应该是这个被点击的view在数组[所有的headerView,用户添加的view,所有的footerView]中的位置(请自行参考HeaderViewListAdapter的getView实现)。而id是来自于adapter.getItemId(position)。

对于ArrayAdapter的getItemId函数,实现就是return position。id和position是一致的。
然而,对于HeaderViewListAdapter

188  public long getItemId(int position) {
189      int numHeaders = getHeadersCount();
190      if (mAdapter != null && position >= numHeaders) {
191          int adjPosition = position - numHeaders;
192          int adapterCount = mAdapter.getCount();
193          if (adjPosition < adapterCount) {
194              return mAdapter.getItemId(adjPosition);
195          }
196      }
197      return -1;
198  }

实现逻辑是,如果position指向了headerView或footerView,那么返回-1,否则,将返回在用户view数组的位置。
也就是说
id=position-headerView的个数(id < headerviewer的个数+用户view的个数),否则=-1
因此,OnItemClickListener的正确实现如下:


1
void onItemClick(AdapterViewparent, View view, int position, long id){
2
    if(id<-1) {
3
        // 点击的是headerView或者footerView
4
        return;
5
    }
6
    int realPosition=(int)id;
7
    T item=getItem(realPosition);
8
    // 响应代码
9
}
原文地址: http://blog.iamzsx.me/show.html?id=147001

Android ListView的OnItemClickListener详解--id和position的区别

  • 0

    开心

    开心

  • 0

    板砖

    板砖

  • 0

    感动

    感动

  • 0

    有用

    有用

  • 0

    疑问

    疑问

  • 0

    难过

    难过

  • 0

    无聊

    无聊

  • 0

    震惊

    震惊

编辑推荐
在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长度自
在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长度自
在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长度自
在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长度自
原文地址:http://www.cnblogs.com/allin/archive/2010/05/11/1732200.html 列表的显示需要三个元素
由于google doc 很多人都打不开,故更新了源码下载地址 【源码下载】----2010-01-18    在android
在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长度自
在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长度自
  在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长
在android开发中ListView是比较常用的组件,它以列表的形式展示具体内容,并且能够根据数据的长度自
版权所有 IT知识库 CopyRight © 2009-2015 IT知识库 IT610.com , All Rights Reserved. 京ICP备09083238号