Android 数据绑定解耦DataBinding的简单使用

来源:互联网 发布:艾博网络 编辑:程序博客网 时间:2024/05/22 02:30

本文将介绍android SDK中自带的框架DataBinding。

使用场景
此框架主要用于数据和View的绑定。例如我们之前写的RecyclerView的Adapter。在自定义的adapter中的onBindViewHolder()方法中通常会根据数据源给view赋值。如:

原始写法
首先在onCreateView中反射出item对应的布局。

 @Override    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.recyclerview_item,parent,false);        MyViewHolder viewHolder = new MyViewHolder(view);               return viewHolder;    }

在把数据赋值给布局中的view

@Override    public void onBindViewHolder(MyViewHolder holder, final int position) {        holder.textView.setText(stringList.get(position));        holder.textView.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                onItemClickListener.onItemClick(position);            }        });    }

使用前在build中做如下配置,注意最外层是android

android {    ...    dataBinding {        enabled = true    }    ...}

而采用框架是怎么写的呢?

@Override    public SampleViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {        RowSampleBinding binding = RowSampleBinding.inflate(LayoutInflater.from(parent.getContext()), parent, false);        return new SampleViewHolder(binding.getRoot());    }
public class SampleViewHolder extends RecyclerView.ViewHolder {        RowSampleBinding binding;        public SampleViewHolder(View itemView) {            super(itemView);            binding = DataBindingUtil.bind(itemView);        }    }

可以看出采用框架后,并没有给控件赋值,其实赋值操作是在绑定数据后,根据xml中的属性自动赋值。上面的代码中首先是构建了一个RowSampleBinding对象,然后用此RowSampleBind构建ViewHolder。直接返回了一个SampleViewHolder对象。这部分代码很好理解,而关键就是这个RowSampleBinding是怎么来的呢?

RowSampleBinding的来源
其实他是自动生成的,是根据row_sample.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"    xmlns:tools="http://schemas.android.com/tools">    <data>        <variable          name="sample"                                type="com.example.yeliang.animation.activityanim.Sample">         </variable>    </data>    <LinearLayout        android:id="@+id/ll_sample"        android:layout_width="match_parent"        android:layout_height="wrap_content"        android:gravity="center_vertical"        android:padding="@dimen/space_double">        <ImageView            android:id="@+id/iv_sample"            style="@style/MaterialAnimationsIconSmall"            android:src="@drawable/circle_24dp"            app:colorTint="@{sample.color}" />        <TextView            android:id="@+id/tv_sample"            style="@style/Base.TextAppearance.AppCompat.Large"            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:text="@{sample.name}"            tools:text="转场动画" />    </LinearLayout></layout>

要看这个布局,还要看和这个布局对应的Bean

public class Sample implements Serializable {    private int color;    private String name;    public Sample(int color, String name) {        this.color = color;        this.name = name;    }    @BindingAdapter("bind:colorTint")    public static void    setColorTint(ImageView view, int color) {        DrawableCompat.setTint(view.getDrawable(), color);    }    public int getColor() {        return color;    }    public String getName() {        return name;    }}

好,接下来分析上面布局。首先看这个标签

<data>        <variable            name="sample"            type="com.example.yeliang.animation.activityanim.Sample">         </variable>  </data>

添加了这个data标签后,就会生成RowSampleBinding
再来看属性,name先不看,先看type。一眼就能看出来,这个type表示此布局对应的Bean的全类名。
那name表示什么呢?name其实就是表示的Bean的名字。而如下:

<ImageView            android:id="@+id/iv_sample"            style="@style/MaterialAnimationsIconSmall"            android:src="@drawable/circle_24dp"            app:colorTint="@{sample.color}" />

可以看到在ImageView控件中,有这一句

app:colorTint="@{sample.color}" 

看到这里就明白了。这个sample就表示Sample的对象,通过上面这一句就会sample.color就会给ImageView设置颜色。当然了这个设置颜色还是通过Sample类中的这个方法:

@BindingAdapter("bind:colorTint")    public static void    setColorTint(ImageView view, int color) {        DrawableCompat.setTint(view.getDrawable(), color);    }

其实这个方法也可以不写,可以直接:
在ImageView中添加下面的这个属性

android:tint="@{sample.color}"

那么数据Bean和布局什么时候绑定到一起的呢?
在RecyclerAdapter中的onBindViewHolder()中

  @Override    public void onBindViewHolder(SampleViewHolder holder, final int position) {        final Sample sample = samples.get(position);        holder.binding.setSample(sample);    }

通过给这一步骤就实现了布局和数据的绑定。

原创粉丝点击