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

以下内容为原创,欢迎转载,转载请注明

来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/4146512.html 

 

给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.ab__id_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.ab__id_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可以依赖AndroidBucket(https://github.com/wangjiegulu/AndroidBucket)项目

上一篇:Servlet第一篇【介绍Servlet、HTTP协议、WEB目录结构、编写入门Servlet程序、Servlet生命周期】(上)


下一篇:【Android游戏开发之四】Android 游戏框架(一个游戏角色在屏幕行走的demo)