Databinding概述:
数据绑定库是一种支持库,借助该库,您可以使用声明性格式(而非程序化地)将布局中的界面组件绑定到应用中的数据源。
布局通常是使用调用界面框架方法的代码在 Activity 中定义的。例如,以下代码调用 findViewById() 来查找 TextView 微件并将其绑定到 viewModel 变量的 userName 属性:
findViewById<TextView>(R.id.sample_text).apply {
text
= viewModel.userName
}
以下示例展示了如何在布局文件中使用数据绑定库将文本直接分配到微件。这样就无需调用上述任何 Java 代码。请注意赋值表达式中 @{} 语法的使用:
<TextViewandroid:text="@{viewmodel.userName}" />
借助布局文件中的绑定组件,您可以移除 Activity 中的许多界面框架调用,使其维护起来更简单、方便。还可以提高应用性能,并且有助于防止内存泄漏以及避免发生 Null 指针异常。
使用数据绑定库
要了解如何在 Android 应用中使用数据绑定库,请参阅以下页面。
了解如何准备开发环境以使用数据绑定库,包括在 Android Studio 中支持数据绑定代码。
借助表达式语言,您可以编写将变量关联到布局中的视图的表达式。数据绑定库会自动生成将布局中的视图与您的数据对象绑定所需的类。该库提供了可在布局中使用的导入、变量和包含等功能。
该库的这些功能可与您的现有布局无缝地共存。例如,可以在表达式中使用的绑定变量在 data 元素(界面布局根元素的同级)内定义。这两个元素都封装在 layout 标记中,如以下示例所示:
<layout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"><data><variablename="viewmodel"type="com.myapp.data.ViewModel" /></data><ConstraintLayout... /> <!-- UI layout's root element --></layout>
数据绑定库提供了可让您轻松地观察数据更改情况的类和方法。您不必操心在底层数据源发生更改时刷新界面。您可以将变量或其属性设为可观察。借助该库,您可以将对象、字段或集合设为可观察。
数据绑定库可以生成用于访问布局变量和视图的绑定类。此页面展示了如何使用和自定义所生成的绑定类。
每一个布局表达式都有一个对应的绑定适配器,要求必须进行框架调用来设置相应的属性或监听器。例如,绑定适配器负责调用 setText() 方法来设置文本属性,或者调用 setOnClickListener() 方法向点击事件添加监听器。最常拥的绑定适配器(例如针对本页面的示例中使用的 android:text 属性)可供您在 android.databinding.adapters 软件包中使用。如需常用绑定适配器列表,请参阅适配器。您也可以按照以下示例所示创建自定义适配器:
@BindingAdapter("app:goneUnless")fun goneUnless(view: View, visible: Boolean) {
view
.visibility = if (visible) View.VISIBLE else View.GONE
}
Android 支持库包含架构组件,您可以使用这些组件设计稳健、可测试且易维护的应用。您可以将架构组件与数据绑定库一起使用,以进一步简化界面开发。
数据绑定库支持双向数据绑定。此类绑定使用的表示法支持以下操作:接收对属性的数据更改,同时监听用户对此属性的更新。
当然以上都是在谷歌开发者官方网站都可以找到,需要的同学可以到官网查看谷歌官方传送门
下面就开始我门今天的示例操作:
首先我门的前期工作就是创建相应的activity和viewModel以及布局文件,当然这个和前面的将Livedata时候的文件基本相同,这里就不在做过多的说明了,想要的同学可以看前面的文章,或者在文章结束的时候,我会给大家我的项目示例的GitHub地址,需要的同学可以下载。
首先来的我先将databinding在我门的项目中启用,这很简单,只需要在我门的gradle中添加以下代码就可以了
dataBinding{
enabled=true
}
设置完成之后,我们同步gradle就可以使用dataBinding了。
首先来到布局文件,在布局文件中的最上面的有一个小黄色的灯泡,点击灯泡,就可以看到一个转换为databinding布局文件的提示;
转换完成之后的布局文件样式如下:
就会发现根布局文件变成了layout了,并多了一个data包裹的区域。
完成布局文件的转换之后,来到activity中进行dataBinding的绑定。绑定如下
public class DataBindActivity extends AppCompatActivity {
private DataBindingViewModel dataBindingViewModel;
//设置dataBinding
private ActivityDataBindBinding dataBindBinding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//绑定不布局文件
dataBindBinding=DataBindingUtil.setContentView(this,R.layout.activity_data_bind);
dataBindingViewModel=new ViewModelProvider(this).get(DataBindingViewModel.class);
//绑定viewModel数据层,setData跟布局文件中的name=data相同
dataBindBinding.setData(dataBindingViewModel);
//生命周期监控
dataBindBinding.setLifecycleOwner(this);
}
}
经过绑定之后我门能够发现,比起之前activity中的代码量少了很多,看起来更加的舒服了。
然后我到布局文件中进行数据和点击事件的绑定代码如下:
<?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">
name="data"
type="com.example.jetpackdemo.dataBinding.DataBindingViewModel" />
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".dataBinding.DataBindActivity">
android:id="@+id/guideline"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical"
app:layout_constraintGuide_begin="205dp" />
android:id="@+id/btn1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="按钮1"
android:onClick="@{()->data.addTickadd1()}"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/guideline"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
android:id="@+id/btn2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="按钮2"
android:onClick="@{()->data.addTickadd2()}"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="@+id/guideline"
app:layout_constraintTop_toTopOf="parent" />
android:id="@+id/tv1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{String.valueOf(data.tickadd1)}"
app:layout_constraintBottom_toTopOf="@+id/btn1"
app:layout_constraintEnd_toStartOf="@+id/guideline"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
android:id="@+id/tv2"
android:layout_width="47dp"
android:layout_height="19dp"
android:text="@{String.valueOf(data.tickadd2)}"
app:layout_constraintBottom_toTopOf="@+id/btn2"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="@+id/guideline"
app:layout_constraintTop_toTopOf="parent" />
</layout>
到这里基本就完成了数据的双向绑定了,接下来看看效果吧!
可以看到实现了我们的目的。
下面是本项目的GitHub地址