[Android]在Adapter的getView方法中绑定OnClickListener比较好的方法


给ListView中每个item绑定点击事件的方法,比较常见的如下这种方式:


 1 public View getView(int positon, View convertView, ViewGroup parent){
2 if(null == convertView){
3 convertView = LayoutInflater.from(context).inflate(R.layout.item, null);
4 }
5
6 Button button = ABViewUtil.obtainView(convertView, R.id.item_btn);
7 button.setOnClickListener(new View.OnClickListener(){
8 @Override
9 public void onClick(View v){
10 Toast.makeText(context, position: + position, Toast.LENGTH_SHORT).show();
11 }
12 });
13
14 }

然后运行,当然没问题。

但是这里有一个可以优化的地方,注意代码第7行,每次调用getView方法都会设置Button的OnClickListener,会生成很多不必要的OnClickListener对象。

所以,我们可以想到,在生成convertView时,同时设置Button的OnClickListener,convertView是被不断地复用的,这样的OnClickListener也就可以被不断地服用,也就是说在第3行和第4行之间进行这一步。这样,代码演化到如下:


 1 public View getView(int positon, View convertView, ViewGroup parent){
2 if(null == convertView){
3 convertView = LayoutInflater.from(context).inflate(R.layout.item, null);
4 Button button = ABViewUtil.obtainView(convertView, R.id.item_btn);
5 button.setOnClickListener(new View.OnClickListener(){
6 @Override
7 public void onClick(View v){
8 Toast.makeText(context, “position: “ + position, Toast.LENGTH_SHORT).show();
9 }
10 });
11 }
12 }

这个代码看上去没什么问题,但是问题就在onClick回调的的第8行中使用的position的时候,因为这里使用的是匿名内部类(这个匿名内部类实现了View.OnClickListener这个接口),所以如果需要在onClick中使用position这个变量的话,需要把position声明为final。一旦声明了final,等到编译之后,这个position就会被作为这个匿名内部类中的一个private的成员变量。这样,ListView往下滚动,下面需要显示的item会重用上面不显示的convertView,convertView中的Button设置的OnClickListener实现类的对象,回调onClick时,使用的position其实是该OnClickListener实现类的成员变量position(这个position的值只是在构造的时候被初始化了而已!)。所以,这个ListView刚加载完数据后,还未滚动时,点击屏幕上的item都是正常的,但是如果一旦滚动,有view被重用了,这个时候,position的值就错乱了,所以在onClick中通过position获取到的item的数据当然也是错乱的了。

所以,要解决这个问题,就需要让onClick方法回调的时候得到的position是个正确的值,我们可以选择使用把当前显示的convertView对应的position值保存在convertView中,然后把这个convertView对象传入OnClickListener中保存,所以代码演化到如下:


 1 public View getView(int positon, View convertView, ViewGroup parent){
2 if(null == convertView){
3 convertView = LayoutInflater.from(context).inflate(R.layout.item, null);
4 Button button = ABViewUtil.obtainView(convertView, R.id.item_btn);
5 button.setOnClickListener(new OnConvertViewClickListener(convertView, R.id.abid_adapter_item_position){
6 @Override
7 public void onClickCallBack(View registedView, int… positionIds){
8 Toast.makeText(context, “position: “ + positionIds[0], Toast.LENGTH_SHORT).show();
9 }
10 });
11 }
12 convertView.setTag(R.id.abid_adapter_item_position, position);
13 }

就像上面第5行这样,Button绑定的是OnConvertViewClickListener,它是OnClickListener的一个实现类可以保存convertView到Listener中,到时候回调onClick方法的时候可以从保存的convertView中获取到当前显示的item的position(这个position是以tag的方式保存在convertView中的)。

当然,还需要做第12行这一步,它的目的是把当前显示的position保存到convertView中,提供给OnConvertViewClickListener获取当前的position。

这样就ok了,可以在onClickCallBack()方法中进行点击事件的处理了,每个button永远只有一个onClickListener。

注:使用OnConvertViewClickListener可以依赖AndroidBuckethttps://github.com/wangjiegulu/AndroidBucket)项目



来源博客:Wang Jie's Blog's Blog
本文链接:https://blog.wangjiegulu.com/2014/12/05/Android-在Adapter的getView方法中绑定OnClickListener比较好的方法/
版权声明:本博客所有文章除特别声明外,均采用 CC BY 4.0 CN协议 许可协议。转载请注明出处。