前言
入行干活一个多月,真心发现代码复用的重要性。为了能兼顾重复代码复用和清晰的逻辑,必须要用一些的基类和控制标志字段来给相同功能属性的类统一起来,并在基类的基础上拓展各自的特性,我觉得这是程序员必须的职业素养。
很长一段时间,我个人写代码在现在看来都是在反复造轮子的无意义行为,以前每新建一个Activity,Fragment,Adapter都是“原生”的,基本上每一个类都有一大堆又臭又长的代码,其实完全可以把这些类的共性提取出来作为一个公共的父类,在根据不同业务需求继承父类在其基础上个性化独立的子类。
今天我来谈谈Anndroid学习初中级阶段遇到的数据适配器Adapter的问题。
笔者我好长一段时间是看视频学习android的,接触到通用数据适配器的时候,看了鸿洋大神写的CommonAdapter,当时觉得"哇,适配器可以写的这么方便,太棒了",于是当下便尝试自己一步一个脚印把他的适配器写了出来,发现原理也不是太难理解,但还是觉得还要用java代码将一个个文本内容设置到TextView中,当不同属性的文本内容很多时,这实在太恶心了,还是有点不太方便。于是便想到用databinding的数据绑定功能和Adapter结合起来,做一个基于DataBinding的CommonAdapter,以后每一个不同的适配器只需要继承这个基类Adapter,就可以快速完成适配器的构建任务。于是便有了下面这个简短的适配器NewsAdapter。
子类: NewsAdapter.java
public class NewsAdapter extends CommonAdapter<ItemNewsBinding,ResultBean.ListBean> {
public NewsAdapter(Context context, List<ResultBean.ListBean> dataList) {
// 调用父类构造方法
super(context, dataList);
}
@Override
public int getLayoutId() {
// 获取item布局xml的文件id
return R.layout.item_news;
}
@Override
public void bindView(CommonViewHolder viewHolder, int position) {
// 将数据加载进databinding绑定的xml中
viewHolder.bindView.setNewsBean(dataList.get(position));
}
}
其中的方法 getLayoutId() , bindView(CommonViewHolder,int) 是CommonAdapter的抽象函数,它俩在CommonAdapter内部调用,分别是为了 获取布局资源id 、通过databinding为布局设置数据。结合databinding,仅仅通过这几行代码就完成了一个adapter的代码构建了,这展现出了继承通用Adapter的强大威力,是不是很神奇,当然,这对于我们这些菜鸟程序猿来说是很惊艳的事情。其实没有想得那么容易,要先将布局文件xml设置成databinding规定的格式,如下所示。
在布局文件 item_news.xml 中,部分代码
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
// 设置新闻数据的bean
<variable
name="newsBean"
type="com.capton.retrofitdemo.News.ResultBean.ListBean"/>
</data>
...省略代码 ...
<ImageView
android:id="@+id/pic"
app:newsImage="@{newsBean.pic}" //设置网络图片
app:placeHolder="@{@drawable/placeholder}" //设置占位图
android:layout_marginLeft="10dp"
android:layout_gravity="center"
android:layout_width="100dp"
android:layout_height="50dp" />
<LinearLayout
android:layout_gravity="center"
android:orientation="vertical"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:text="@{newsBean.title}" // 设置新闻标题
android:id="@+id/title"
android:textSize="14sp"
android:textColor="@color/black"
android:layout_marginBottom="1dp"
android:gravity="bottom|left"
省略代码...
代码里面关于利用databinding设置网络图片的过程,这里就不提及了,大家看其他前辈们的简书能找到很多文章。
通用适配器 CommonAdapter.java
public abstract class CommonAdapter<DB extends ViewDataBinding,T> extends RecyclerView.Adapter {
protected List<T> dataList;
private Context context;
public List<T> getDataList() {
return dataList;
}
public void setDataList(List<T> dataList) {
this.dataList = dataList;
}
public CommonAdapter(Context context, List<T> dataList) {
this.dataList = dataList;
this.context = context;
}
// 抽象函数 获取布局资源id
public abstract int getLayoutId();
// 抽象函数 通过databinding为布局设置数据
public abstract void bindView(CommonViewHolder viewHolder,int position);
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
// 生成DB对象 (这个方法是不是和View.inflate()很像?)
DB bindView= DataBindingUtil.inflate(LayoutInflater.from(context),getLayoutId(),parent,false);
return new CommonViewHolder(bindView);
}
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position){
// 调用抽象函数,将holder强转为CommonViewHodler,供子类Adapter使用其成员对象bindView;
bindView((CommonViewHolder) holder,position);
}
@Override
public int getItemCount() {
return dataList.size();
}
@Override
public int getItemViewType(int position) {
return super.getItemViewType(position);
}
public class CommonViewHolder extends RecyclerView.ViewHolder{
public DB bindView;
// 每一个item都必须持有的一个ViewDataBinding子类对象
public CommonViewHolder(DB bindView) {
super(bindView.getRoot());
this.bindView=bindView;
}
}
}
说明
例如,在NewsAdapter.java中,应该这样写
至于这个ItemNewsBinding命名是有规律的,根据Databinding的生成原则:每一个大驼峰单词取自布局文件xml的文件名每一个下划线隔开的单词+Binding,例如:item_news.xml -> ItemNewsBinding
后记
现在这个基于databinding的通用数据适配器还没考虑加载不同数据类型的情况,后面再根据需求改进吧。