Android Databinding 在adapter中完全取代ViewHolder实践

张虾昂 / 文 发表于2015-07-09 09:26 次阅读 adapter,Android Databinding

          在文章的开始感谢@rufi 对我的帮助,积极解答我的一些疑问。大家如果对于databinding还不太了解的话可以去他的github上去了解更多的databinding的应用。

          地址如下: https://github.com/LyndonChin/MasteringAndroidDataBinding   配置的环境在他的文章里都写的非常详细,有疑问的可以直接咨询他哟,如果是比较简单的问题就直接留言在文章的下面吧,我会一一解答的(不懂我就去问他😄)

          有所了解的都知道databinding出现的意义就在于系统自动帮助我们生成Data和View之间的connection.而在Adapter中出现的ViewHolder就是我们手动生成的connection,下面我们就来看看databinding是如何完全取代ViewHolder的角色吧。(在此之前请看如何配置环境,在上述链接里有)

  

public class DynamicActivity extends Activity {
   private View.OnClickListener itemClickListener;
   private GridView gridView;

   @Override
   protected void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_dynamic);
       itemClickListener = new View.OnClickListener() {
           @Override
           public void onClick(View v) {
               Toast.makeText(getBaseContext(), v.getClass().getSimpleName(), Toast.LENGTH_SHORT).show();
           }
       };
       gridView = (GridView) findViewById(R.id.gridView);
       MMyAdapter adapter = new MMyAdapter(getBaseContext());
       adapter.setOnclickListener(itemClickListener);
       gridView.setAdapter(adapter);
   }
}

 


         listener用于处理item的一些click事件,当然也可以直接写到adapter里面去,但是考虑到adapter还是比较纯粹的处理view和data的关系,所以把这个还是放到了activity层。

        接着是activity_main.xml文件比较简单就一个gridview


<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
   
android:layout_width="match_parent"
   
android:layout_height="match_parent">
   <GridView
       
android:id="@+id/gridView"
       
android:layout_width="match_parent"
       
android:layout_height="wrap_content"
       
android:numColumns="3" />
</RelativeLayout>


   接着是比较关键的adapter


/**
* Created by zhangxiaang on 15/7/6.
*/
public class MMyAdapter extends BaseAdapter {
   private Context mContext;
   private LayoutInflater inflater;
   private List<String> mlist;
   public View.OnClickListener itemClickListener;
   private ItemmBinding binding;

   public MMyAdapter(Context context) {
       this.mContext = context;
       inflater = LayoutInflater.from(context);
       mlist = new ArrayList<String>();
       for (int i = 0; i < 100; i++) {
           mlist.add(i, "item" + i);
       }
   }
   @Override
   public int getCount() {
       return mlist == null ? 0 : mlist.size();
   }

   @Override
   public Object getItem(int position) {
       return mlist.get(position);
   }

   @Override
   public long getItemId(int position) {
       return position;
   }

   @Override
   public View getView(int position, View convertView, ViewGroup parent) {
       if (convertView == null) {
           binding = DataBindingUtil.inflate(inflater, R.layout.itemm, parent, false);
           convertView = binding.getRoot();
           convertView.setTag(binding);
       } else {
           binding = (ItemmBinding) convertView.getTag();
       }
       binding.setVariable(BR.item, mlist.get(position));
       binding.setAdapter(this);
       return convertView;
   }
   public void setOnclickListener(View.OnClickListener itemClickListener) {
       this.itemClickListener = itemClickListener;
   }
}

其中的item布局如下:


<layout xmlns:android="http://schemas.android.com/apk/res/android"
   
xmlns:app="http://schemas.android.com/apk/res-auto">

   <data>
       <variable
           
name="item"
           
type="String" />
       <variable
           
name="adapter"
           
type="com.liangfeizc.databindingsamples.dynamic.MMyAdapter" />
   </data>

   <LinearLayout
       
android:layout_width="match_parent"
       
android:layout_height="match_parent"
       
android:orientation="vertical">

       <TextView
           
android:id="@+id/textView"
           
android:layout_width="wrap_content"
           
android:layout_height="wrap_content"
           
android:clickable="true"
           
app:text="@{item}"
           
app:onClickListener="@{adapter.itemClickListener}"/>
   </LinearLayout>
</layout>


可以看到的是在adapter中完全没有 viewHolder的身影,在getView中,开始直接判断当前的convertview是否为空,进而来初始化这个binding对象,而convertView就是这个binding对象所对应的container的root view类似一个view-tree的rootView角色,然后同样的给当前convertView打一个标签方便后续取出,如果不为空的话就直接取出当前convertView的binding对象。

处理完convertView的视图部分接着就是处理convertView的数据部分,而数据部分应该在包含在convertview的binding中进行赋值。(感兴趣的可以去看看生成的binding代码,自动的替我们生成了<variable>标签里面数据的setter函数)

 

最后把设置完了view和data的converView直接返回就ok 啦,大家跑一边看看databinding带来的便利吧。

 

Ps:由于在rufi的案例中已经给了一个recyclerView的案例,但是我个人觉得recyclerView中已经封装了ViewHolder,所以再使用databinding的话就显得有点多余了。毕竟databinding的存在意义就是取代掉viewHolder的。

 

欢迎各位提出自己的一些想法或者实现过程中遇到的问题。


收藏 赞 (19) 踩 (1)