Android jetpack - DataBinding 数据绑定库

一、前言

DataBinding 数据绑定库属于谷歌在2018推出Android jetpack(外网)其中的软件架构组件中的一个。在谷歌开发者网站有详细的介绍DataBinding(外网)。本文旨在结合官方文档和官方demo-github-databinding-samples用更加通俗易懂的方式全面深入的解析数据绑定库。

二、简介

套用谷歌文档中的一句话:数据绑定库是一种支持库,借助该库,您可以使用声明性格式(而非程序化地)将布局中的界面组件绑定到应用中的数据源。
以前我们写代码是这样的:

TextView textView = findViewById(R.id.sample_text);
    textView.setText(viewModel.getUserName());

用数据绑定可以直接在布局就定义了,省略了上面的代码。布局中代码是这样的:

<TextView
        android:text="@{viewmodel.userName}" />

数据绑定库的核心就是这个了。数据变化时,都不用再次set。

三、使用DataBinding

会从以下几个点来说明如何使用DataBinding:1、使用入门 2、布局绑定表达式 3、使用可观察的数据对象 4、生成的绑定类 5、绑定适配器 6、将布局和视图绑定到架构视图 7、双向绑定数据。

3.1、使用入门

首先硬性条件:DataBinding支持Android 4.0 (API级别14)或更高。com.android.tools.build:gradle1.5.0或更高。
<1>、配置

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

<2>、把鼠标点到布局上面,出现黄色灯泡的时候,点击灯泡。转换布局为databinding。


图片.png

自动转换成下面的样子,

<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>

   </data>
   <androidx.constraintlayout.widget.ConstraintLayout
           android:layout_width="match_parent"
           android:layout_height="match_parent">

       <TextView
...

<3>、加上数据

    <data>
        <variable name="name" type="String"/>
        <variable name="buttonName" type="String"/>
    </data>
    <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@{name}" />
    <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@{buttonName}"

<4>、最后一步绑定数据

    ActivityMainBinding activityMainBinding;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //setContentView(R.layout.activity_main);
        activityMainBinding = DataBindingUtil.setContentView(this,R.layout.activity_main);
        activityMainBinding.setName("jack");
        activityMainBinding.setButtonName("button");
    }

    public void onClick(View view) {
        activityMainBinding.setName("yink");
    }

至此一个简单的应用就出来了,效果就是我们只需要更改binding的数据,view上对应会自动去更新。
优点
1、您可以移除 Activity 中的许多界面框架调用,使其维护起来更简单、方便。
2、提高应用性能,并且有助于防止内存泄漏以及避免空指针异常。

3.2、布局绑定表达式

3.2.1、如何绑定

数据绑定库会为每一个布局自动生成将布局中的视图与数据对象绑定所需的类。默认会以布局的名字以驼峰命名法加尾缀来取名ActivityMainDataBinding。类包含从布局属性(例如,用户变量)到布局视图的所有绑定,并且知道如何为绑定表达式赋值。
下面举一个自定义数据类的例子

public class SimpleViewModel {
    String name;
    String age;
    SimpleViewModel(String name,String age){
        this.name = name;
        this.age = age;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setAge(String age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public String getAge() {
        return age;
    }

    public String name() {
        return "honny";
    }
}
图片.png

在我们使用的时候:
1、simpleViewModel.name默认是调用的SimpleViewModel的getName方法,如果未定义getName方法不能使用simpleViewModel.name。
2、可以自己定义一些别的方法,比如 public String name() 也是直接可以调用。
最后绑定数据

activityMainBinding = DataBindingUtil.setContentView(this,R.layout.activity_main);
simpleViewModel = new SimpleViewModel("a","18");
activityMainBinding.setSimpleViewModel(simpleViewModel);

更改数据,单独更改数据不会刷新view,本文会在3.3节讲解一种方式,每当你更改数据databinding会自动帮我们去刷新UI,但是现在你只能通过调用activityMainBinding.setSimpleViewModel或者invalidateAll来达到刷新的目的

simpleViewModel.setName("123");
activityMainBinding.setSimpleViewModel(simpleViewModel);
//activityMainBinding.invalidateAll();

另外,数据绑定也十分灵活,和我们平时加载各种布局类似

ActivityMainBinding binding = ActivityMainBinding.inflate(getLayoutInflater());
ListItemBinding binding = ListItemBinding.inflate(layoutInflater, viewGroup, false);
ListItemBinding binding = DataBindingUtil.inflate(layoutInflater, R.layout.list_item, viewGroup, false);

此处注意两个点
1、布局写错了对应的dataBinding实现可能找不到,如果你在library里边用了databinding,导入到项目主build.gradle里边也要加dataBinding {enabled = true} 声明。
2、通常inflate加载的方式需要返回view视图,比如fragment,你需要如下代码

View root = fragmentAppInfoBinding.getRoot();
return root;

3.2.2、运算符

在布局中,我们还可以使用运算符来简化一些表达式


    算术运算符 + - / * %
    字符串连接运算符 +
    逻辑运算符 && ||
    二元运算符 & | ^
    一元运算符 + - ! ~
    移位运算符 >> >>> <<
    比较运算符 == > < >= <=(请注意,< 需要转义为 &lt;)
    instanceof
    分组运算符 ()
    字面量运算符 - 字符、字符串、数字、null
    类型转换
    方法调用
    字段访问
    数组访问 []
    三元运算符 ?:

一个简单的例子如下:

<data>
        <variable name="name" type="String"/>
        <import type="android.view.View"/>
    </data>

<TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@{String.format(@string/copy_right,appInfo.appName)}"
            android:padding="@{large? @dimen/largePadding : @dimen/smallPadding}"
            android:visibility="@{age>13?View.GONE:View.VISIBLE}"

字符串可以”“中加‘’来使用,也可以‘’中加”“来使用
可以import导入一些你想要使用的类,除常规类型使用都必须导入。
资源/表达式等可以直接使用
格式字符串(字符串资源有引用)和复数形式可通过提供参数进行求值

Null 合并运算符,等价于第二个用法

android:text="@{user.displayName ?? user.lastName}"
android:text="@{user.displayName != null ? user.displayName : user.lastName}"

您可以在托管代码中使用的表达式语法中缺少以下运算:
this、super、new、显式泛型调用

集合

为方便起见,可使用 [] 运算符访问常见集合,例如数组、列表、稀疏列表和映射。
其中还可指定变量类型,例如type="List<String>"

<data>
        <import type="android.util.SparseArray"/>
        <import type="java.util.Map"/>
        <import type="java.util.List"/>
        <variable name="list" type="List&lt;String>"/>
        <variable name="sparse" type="SparseArray&lt;String>"/>
        <variable name="map" type="Map&lt;String, String>"/>
        <variable name="index" type="int"/>
        <variable name="key" type="String"/>
    </data>
    …
    android:text="@{list[index]}"
    …
    android:text="@{sparse[index]}"
    …
    android:text="@{map[key]}"
    

事件

1、方法引用
你可以直接指定onClick,这种是直接引用实例的方法

<ImageButton
            android:id="@+id/back_ibt"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:onClick="@{appInfoFragment.onClick}"/>

2、监听器绑定
监听器绑定用lambda表达式(JDK8新特性)来表述。
lambda:语法形式为 () -> {},其中 () 用来描述参数列表,{} 用来描述方法体,-> 为 lambda运算符 ,读作(goes to)。
其中参数列表可以全部忽略,忽略写法如下

public class Presenter {
        public void onSaveClick(Task task){}
    }
<Button android:layout_width="wrap_content" android:layout_height="wrap_content"
            android:onClick="@{() -> presenter.onSaveClick(task)}" />

上面表达式可下面这种写法等价

android:onClick="@{(view) -> presenter.onSaveClick(task)}"

3.3、使用可观察数据对象

文中前边讲解的都是如何绑定数据、设置监听以及语法等。我们还需要灵活的刷新UI,不用我没每次手动去设置更新。databinding给我们提供了三个类别的监听刷新。“对象、字段、集合”
实现 Observable 接口的类允许注册监听器,以便它们接收有关可观察对象的属性更改的通知。于是每当我们更改数据的时候,databinding会自动去刷新UI
字段

ObservableBoolean
ObservableByte
ObservableChar
ObservableShort
ObservableInt
ObservableLong
ObservableFloat
ObservableDouble
ObservableParcelable
private static class User {
        public final ObservableField<String> firstName = new ObservableField<>();
        public final ObservableField<String> lastName = new ObservableField<>();
        public final ObservableInt age = new ObservableInt();
    }
user.firstName.set("Google");
    int age = user.age.get();

集合

ObservableArrayMap<String, Object> user = new ObservableArrayMap<>();
    user.put("firstName", "Google");
    user.put("lastName", "Inc.");

<TextView
        android:text="@{String.valueOf(1 + (Integer)user.age)}"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

ObservableArrayList<Object> user = new ObservableArrayList<>();
    user.add("Google");
    user.add("Inc.");
    user.add(17);

对象

 private static class User extends BaseObservable {
        private String firstName;
        private String lastName;

        @Bindable
        public String getFirstName() {
            return this.firstName;
        }

        @Bindable
        public String getLastName() {
            return this.lastName;
        }

        public void setFirstName(String firstName) {
            this.firstName = firstName;
            notifyPropertyChanged(BR.firstName);
        }

        public void setLastName(String lastName) {
            this.lastName = lastName;
            notifyPropertyChanged(BR.lastName);
        }
    }

User是对象,我们引用User当数据时,User对象不能改变,它是同一个对象。俗称数据基类无法改变。为了有效的监听databinding采用如下处理方式:
databinding会生成一个BR的类,BR包含数据绑定资源的ID。调用notifyPropertyChanged方法可以有效的注册和通知监听。

3.4、生成绑定类

绑定类的作用就是让布局视图和布局变量关联起来。系统会为每个布局产生一个绑定类。

  • 下面列举各种绑定方式
MyLayoutBinding binding = MyLayoutBinding.inflate(getLayoutInflater());

MyLayoutBinding binding = MyLayoutBinding.inflate(getLayoutInflater(), viewGroup, false);

MyLayoutBinding binding = MyLayoutBinding.bind(viewRoot);

View viewRoot = LayoutInflater.from(this).inflate(layoutId, parent, attachToParent);
ViewDataBinding binding = DataBindingUtil.bind(viewRoot);

ListItemBinding binding = ListItemBinding.inflate(layoutInflater, viewGroup, false);
// or
ListItemBinding binding = DataBindingUtil.inflate(layoutInflater, R.layout.list_item, viewGroup, false);
  • 视图控件有id,数据绑定库为为其创建不可变字段

  • 变量,数据绑定库为自动生成setter,getter方法

  • 可变对象发生改变时,默认下一帧发生改变;但有时必须手动executePendingBindings()强制执行

  • 指定绑定类路径<data class="com.example.ContactItem"> … </data>

  • 动态变量 ,不知道特定的绑定类。比如recycleview

public void onBindViewHolder(BindingHolder holder, int position) {
        final T item = items.get(position);
        holder.getBinding().setVariable(BR.item, item);
        holder.getBinding().executePendingBindings();
    }

3.5、绑定适配器

3.5.1、适配器

自动选择适配器
首先理解安卓自带的适配器,比如android:text="abc"实际就是会自动绑定setText的适配器,我们可以自定义TextView时复写setText方法来改变默认给的abc值

@Override
    public void setText(CharSequence text, BufferType type) {
        super.setText(text+"aa", type);
    }

这种平时我们使用的方式就叫做自动选择适配器

控制适配器
DataBinding库给我们灵活控制这种绑定适配器的操作。

@BindingMethods({@BindingMethod(type = TextView.class,attribute = "android:text",method = "setTextAddTail")})
public class MyTextView extends TextView {
    ...
    public void setTextAddTail(String text){
        this.setText(text+"_tail");
    }
}

这样,我们就可以更改默认的android:text的适配器为我们自己写的public void setTextAddTail(String text)方法

提供适配器逻辑
1、有的属性没有适配器的,比如android:paddingLeft这个时候我们可以给控件写个适配器

@BindingAdapter("android:paddingLeft")
    public static void setPaddingLeft(View view, int padding) {
      view.setPadding(padding,
                      view.getPaddingTop(),
                      view.getPaddingRight(),
                      view.getPaddingBottom());
    }

2、还可以多个属性同时用,比如同时指定imageUrlerror两个属性才会调用下面的适配器

@BindingAdapter({"imageUrl", "error"})
    public static void loadImage(ImageView view, String url, Drawable error) {
      Picasso.get().load(url).error(error).into(view);
    }

当然也可以多个属性时,添加requireAll=false字段,只指定其中一个属性时也生效

@BindingAdapter(value={"imageUrl", "placeholder"}, requireAll=false)
    public static void setImageUrl(ImageView imageView, String url, Drawable placeHolder) {
      if (url == null) {
        imageView.setImageDrawable(placeholder);
      } else {
        MyImageLoader.loadInto(imageView, url, placeholder);
      }
    }

3、如果是事件的适配器,必须是具有一种抽象方法的接口或抽象类一起使用

<View android:onLayoutChange="@{() -> handler.layoutChanged()}"/>

@BindingAdapter("android:onLayoutChange")
    public static void setOnLayoutChangeListener(View view, View.OnLayoutChangeListener oldValue,
           View.OnLayoutChangeListener newValue) {
      if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
        if (oldValue != null) {
          view.removeOnLayoutChangeListener(oldValue);
        }
        if (newValue != null) {
          view.addOnLayoutChangeListener(newValue);
        }
      }
    }

监听器也可指定单个生效

@BindingAdapter({"android:onViewDetachedFromWindow", "android:onViewAttachedToWindow"}, requireAll=false)
    public static void setListener(View view, OnViewDetachedFromWindow detach, OnViewAttachedToWindow attach) {

3.5.2、对象自动转换

DataBinding库在引用变量时,若参数是Object,会自动转换匹配的类型

<TextView
       android:text='@{userMap["lastName"]}'

这里会自动转换为String

自定义转换

<View
       android:background="@{isError ? @color/red : @color/white}"
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"/>

这里的background需要的是Drawable才对,所以需要我们自己去写个方法来转换它

@BindingConversion
    public static ColorDrawable convertColorToDrawable(int color) {
        return new ColorDrawable(color);
    }

提供了转换方法,在xml中写时也必须保持参数一致,@{isError ? @drawable/error : @color/white}这样是不允许的

3.6、 架构组件中的数据

现在布局中我们可以自由的添加数据,创建自己的数据,也可以方便的来回监听。现在我们使用AndroidX中其它组件,例如LiveData数据等,接下来我们看像LiveData这类的数据如何和我们的布局绑定。
下面的例子用到了LiveData和ViewModel。这里简单说下
LiveData:具有生命周期感知能力的数据存储类。可在适当生命周期及时监听。简单理解为数据存储封装。
ViewModel:在以注重生命周期的方式存储和管理界面相关的数据。它的用途是封装界面控制器的数据。

public class MyViewModel extends ViewModel {
    MutableLiveData<String> name;

    public MutableLiveData<String> getName() {
        if (name == null) {
            name = new MutableLiveData<>();
        }
        return name;
    }
}

        <import type="com.example.demo.MyViewModel"/>
        <variable name="viewModel" type="MyViewModel" />

<TextView
            android:id="@+id/test_tv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text='@{viewModel.name}'
            />

ViewModelProvider.Factory factory = ViewModelProvider.AndroidViewModelFactory.getInstance(this.getApplication());
        ViewModelProvider viewModelProvider = new ViewModelProvider(this, factory);
        viewModel = viewModelProvider.get(MyViewModel.class);
        viewModel.getName().observe(this, new Observer<String>() {
            @Override
            public void onChanged(String s) {
                textView.setText(s);
            }
        });

MyViewModel我们自定义的页面数据管理,它管理一个LiveData封装的String,数据绑定库的数据源只是变成了LiveData,MyViewModel更像一个中介。
这样布局视图和架构组件就结合在了一起。说简单一点就是数据用的AndroidX中提供的组件来写。

3.7、双向绑定

以最通俗的例子,讲解双向绑定。比如一个EditTextView,我绑定了一个LiveData的String。每当string改变时,editTextView就刷新显示了。这是我们数据绑定库最基本的用法。现在我点击EditTextView去更改了里边的内容,这个时候,界面刷新一下,还是变成了LiveData之前保存的String。所以我们需要双向绑定。当更改了editTextView中的值的时候,对应的也吧LiveData中的值修改了。这就是双向绑定。

<CheckBox
        android:id="@+id/rememberMeCheckBox"
        android:checked="@={viewmodel.rememberMe}"
    />

public class LoginViewModel extends BaseObservable {
        // private Model data = ...

        @Bindable
        public Boolean getRememberMe() {
            return data.rememberMe;
        }

        public void setRememberMe(Boolean value) {
            // Avoids infinite loops.
            if (data.rememberMe != value) {
                data.rememberMe = value;

                // React to the change.
                saveData();

                // Notify observers of a new value.
                notifyPropertyChanged(BR.remember_me);
            }
        }
    }
  • 双向绑定表达式@={viewmodel.rememberMe}

四、写在最后

现在再来理解什么是数据绑定库:它是一种支持库,借助该库,您可以使用声明性格式(而非程序化地)将布局中的界面组件绑定到应用中的数据源。
有几点个人看法,不对之处欢迎指正:
优点

  • 代码变得精简、页面逻辑简单、省去了findbyid等代码
    缺点:
  • 错误提示不够友好
  • 有些还是不支持,比如"@{@drawable/list_default}"可以"@{@mipmap/list_default}"不行
  • 布局变得复杂
  • 多模块开发,子library用databinding,主模块也必须设置enable,如果布局名称相同且同时使用databinding,主程序会根据子模块的布局来生成databinding
  • 其它

databinding最大最推崇的就是真的精简了很多,databinding也被应用于MVVM框架。是MVVM的典型用法。也是让UI以数据来驱动的推荐写法。而另一种以数据为核心驱动界面的方法就是2019年谷歌io大会上新推框架,现在已经有预览版的jetpack compose(外网)。DataBinding它确实还不够完美,容易采坑。Google爸爸也在不断的改进,希望数据绑定库越来越好用。

参考文献
DataBinding(外网)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容